Bagaimana untuk lulus pembolehubah dengan rujukan?

PHPz
Lepaskan: 2024-02-09 20:00:05
ke hadapan
555 orang telah melayarinya

Bagaimana untuk lulus pembolehubah dengan rujukan?

Kandungan soalan

Saya menulis kelas ini untuk ujian:

class PassByReference:
    def __init__(self):
        self.variable = 'Original'
        self.change(self.variable)
        print(self.variable)

    def change(self, var):
        var = 'Changed'
Salin selepas log masuk

Apabila saya cuba mencipta contoh, outputnya ialah original。所以python中的参数似乎是按值传递的。那是对的吗?如何修改代码才能达到引用传递的效果,使得输出为changed?

Kadang-kadang orang terkejut dengan x = 1 这样的代码(其中 x 是参数名称)不会影响调用者的参数,但像 x[0] = 1 这样的代码却会影响调用者的参数。发生这种情况是因为尽管有 = suka sintaks, tetapi tugasan item dan tugasan hirisan ialah kaedah mengubahobjek sedia ada, bukannya menetapkan semula pembolehubah. Lihat Mengapa fungsi boleh mengubah suai beberapa parameter seperti yang dilihat oleh pemanggil, tetapi bukan yang lain? Ketahui lebih lanjut.

Juga lihat Apakah perbezaan antara lulus dengan rujukan dan lulus dengan nilai? Untuk perbincangan terminologi yang penting dan bebas bahasa.


Jawapan yang betul

parameter adalah diluluskan oleh tugasan

. Terdapat dua sebab di sebalik ini:
  1. Parameter yang diluluskan sebenarnya adalah rujukan
  2. kepada objek (tetapi rujukan itu diluluskan oleh nilai)
  3. Sesetengah jenis data boleh berubah, tetapi yang lain tidak

Jadi:
  • Jika anda melepasi objek bolehubah

    kepada kaedah, kaedah itu akan mendapat rujukan kepada objek yang sama dan anda boleh mengubahnya mengikut cara yang anda mahu, tetapi jika anda berada di dalam kaedah, skop luar tidak mengetahui apa-apa mengenainya Selepas itu, rujukan luaran masih akan menunjuk ke objek asal.
  • Jika anda menghantar objek tidak berubah

    kepada kaedah, anda masih tidak boleh mengikat semula rujukan luaran atau menukar objek itu.

Untuk menggambarkan perkara ini dengan lebih jelas, mari berikan beberapa contoh.

Senarai - jenis boleh ubah

Mari cuba ubah suai senarai yang diserahkan kepada kaedah:

def try_to_change_list_contents(the_list):
    print('got', the_list)
    the_list.append('four')
    print('changed to', the_list)

outer_list = ['one', 'two', 'three']

print('before, outer_list =', outer_list)
try_to_change_list_contents(outer_list)
print('after, outer_list =', outer_list)
Salin selepas log masuk

Keluaran:

before, outer_list = ['one', 'two', 'three']
got ['one', 'two', 'three']
changed to ['one', 'two', 'three', 'four']
after, outer_list = ['one', 'two', 'three', 'four']
Salin selepas log masuk
outer_listMemandangkan parameter yang diluluskan adalah rujukan kepada

dan bukan salinannya, kita boleh menggunakan kaedah senarai mutasi untuk menukarnya dan perubahan itu ditunjukkan dalam skop luar.

Sekarang mari kita lihat apa yang berlaku apabila kita cuba menukar rujukan yang dihantar sebagai parameter:

def try_to_change_list_reference(the_list):
    print('got', the_list)
    the_list = ['and', 'we', 'can', 'not', 'lie']
    print('set to', the_list)

outer_list = ['we', 'like', 'proper', 'english']

print('before, outer_list =', outer_list)
try_to_change_list_reference(outer_list)
print('after, outer_list =', outer_list)
Salin selepas log masuk

Keluaran:

before, outer_list = ['we', 'like', 'proper', 'english']
got ['we', 'like', 'proper', 'english']
set to ['and', 'we', 'can', 'not', 'lie']
after, outer_list = ['we', 'like', 'proper', 'english']
Salin selepas log masuk
the_list 参数是按值传递的,因此为其分配新列表不会对方法外部的代码产生任何影响。 the_listouter_list 引用的副本,我们让 the_list 指向一个新列表,但无法更改 outer_listMemandangkan parameter the_list diluluskan mengikut nilai, memberikan senarai baharu kepadanya tidak akan memberi kesan pada kod di luar kaedah. the_list ialah salinan daripada apa yang

merujuk, kami menjadikan the_list menghala ke senarai baharu, tetapi kami tidak boleh menukar ke mana

menunjuk ke.

String - jenis tidak berubah

Ia tidak berubah jadi kita tidak boleh menukar kandungan rentetan

Sekarang, mari cuba ubah rujukan

def try_to_change_string_reference(the_string):
    print('got', the_string)
    the_string = 'in a kingdom by the sea'
    print('set to', the_string)

outer_string = 'it was many and many a year ago'

print('before, outer_string =', outer_string)
try_to_change_string_reference(outer_string)
print('after, outer_string =', outer_string)
Salin selepas log masuk

Keluaran: the_string 参数是按值传递的,因此为其分配新字符串不会对方法外部的代码看到任何影响。 the_stringouter_string 引用的副本,我们让 the_string 指向一个新字符串,但无法更改 outer_string

before, outer_string = it was many and many a year ago
got it was many and many a year ago
set to in a kingdom by the sea
after, outer_string = it was many and many a year ago
Salin selepas log masuk

Sekali lagi, kerana di mana

mata.

Saya harap ini menjadikan perkara lebih jelas.

EDIT:

Seseorang menunjukkan bahawa ini tidak menjawab soalan asal @david: "Adakah ada apa-apa yang boleh saya lakukan untuk lulus pembolehubah dengan rujukan sebenar?". Mari bekerja keras.

Bagaimana kita nak selesaikan masalah ini?

Seperti yang ditunjukkan oleh jawapan @andrea, anda boleh mengembalikan nilai baharu. Ini tidak mengubah cara kandungan dihantar, tetapi ia membolehkan anda mendapatkan maklumat yang anda inginkan:

def return_a_whole_new_string(the_string):
    new_string = something_to_do_with_the_old_string(the_string)
    return new_string

# then you could call it like
my_string = return_a_whole_new_string(my_string)
Salin selepas log masuk

Jika anda benar-benar ingin mengelak daripada menggunakan nilai pulangan, anda boleh mencipta kelas untuk menyimpan nilai anda dan menghantarnya kepada fungsi atau menggunakan kelas sedia ada, seperti senarai:

def use_a_wrapper_to_simulate_pass_by_reference(stuff_to_change):
    new_string = something_to_do_with_the_old_string(stuff_to_change[0])
    stuff_to_change[0] = new_string

# then you could call it like
wrapper = [my_string]
use_a_wrapper_to_simulate_pass_by_reference(wrapper)

do_something_with(wrapper[0])
Salin selepas log masuk
🎜Walaupun ini kelihatan agak menyusahkan. 🎜

Atas ialah kandungan terperinci Bagaimana untuk lulus pembolehubah dengan rujukan?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Label berkaitan:
sumber:stackoverflow.com
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan
Tentang kita Penafian Sitemap
Laman web PHP Cina:Latihan PHP dalam talian kebajikan awam,Bantu pelajar PHP berkembang dengan cepat!