""" # 1 Python的函數參數傳遞 這里記住的是類型是屬於對象的,而不是變量。 而對象有兩種,“可更改”(mutable)與“不可更改”(immutable)對象。在python中,strings, tuples, 和numbers是不可更改的對象,而 list, dict, set 等則是可以修改的對象。(這就是這個問題的重點) 當一個引用傳遞給函數的時候,函數自動復制一份引用,這個函數里的引用和外邊的引用沒有半毛關系了. 所以第一個例子里函數把引用指向了一個不可變對象,當函數返回的時候,外面的引用沒半毛感覺.而第二個例子就不一樣了, 函數內的引用指向的是可變對象,對它的操作就和定位了指針地址一樣,在內存里進行修改. """ # a = 1 # def fun(a): # print( "func_in",id(a)) # func_in 1604579120 # a = 2 # print( "re-point",id(a), id(2)) # re-point 1604579152 1604579152 # # print( "func_out",id(a), id(1)) # func_out 1604579120 1604579120 # fun(a) # print (a) # 1 """ 輸出 func_out 1604579120 1604579120 func_in 1604579120 re-point 1604579152 1604579152 1 Process finished with exit code 0 """ # 所有的變量都可以理解是內存中一個對象的“引用” # 可以看到,在執行完a = 2之后,a引用中保存的值,即內存地址發生變化,由原來1對象的所在的地址變成了2這個實體對象的內存地址。
""" 參數通過賦值傳遞。這背后的理由是雙重的: 傳遞的參數實際上是一個參考的一個對象(但參考通過值傳遞) 一些數據類型是可變的,但其他數據類型則不可變 所以: 如果你將一個可變對象傳遞給一個方法,那么該方法會獲得對同一個對象的引用,你可以將它改變,但是如果你在方法中重新引用引用,那么外部范圍將對它一無所知,之后你完成后,外部引用仍將指向原始對象。 如果將不可變對象傳遞給方法,則仍然無法重新綁定外部引用,甚至無法改變對象。 為了更清楚,讓我們舉一些例子。 """ """ 列表 - 可變類型 讓我們嘗試修改傳遞給方法的列表: """ # 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) """ 輸出 before, outer_list = ['one', 'two', 'three'] got ['one', 'two', 'three'] changed to ['one', 'two', 'three', 'four'] after, outer_list = ['one', 'two', 'three', 'four'] """ # 由於傳入的參數是outer_list對它的引用,而不是它的副本,我們可以使用變異列表方法來更改它並使更改反映在外部作用域中。
""" 現在讓我們看看當我們嘗試更改作為參數傳入的引用時會發生什么: """ # 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) """ 輸出 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'] """ # 由於the_list參數是按值傳遞的,因此為其分配新列表不會影響方法外部的代碼。這the_list是outer_list引用的副本,我們the_list指向了一個新列表,但沒有辦法改變outer_list指向的位置
""" 字符串 - 不可變類型 它是不可變的,所以我們無法改變字符串的內容 現在,讓我們嘗試更改引用 """ # 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) # """ 輸出 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 """ # 同樣,由於the_string參數是通過值傳遞的,因此為其分配新字符串不會影響方法外部的代碼。這the_string是一個outer_string引用的副本,我們the_string指向一個新的字符串,但沒有辦法改變outer_string指向的位置。
a = 1 a = 2 """ 您認為這a是存儲值的內存位置1,然后更新以存儲該值2。這不是Python中的工作方式。 相反,a作為對具有該值的對象的引用開始1,然后被重新分配為具有該值的對象的引用2。 這兩個對象可能會繼續共存,即使a不再引用第一個對象; 實際上,它們可能被程序中的任何其他引用共享。 當您使用參數調用函數時,會創建一個引用傳入對象的新引用。這與函數調用中使用的引用是分開的,因此無法更新該引用並使其引用新對象。在你的例子中: """ # def __init__(self): # self.variable = 'Original' # self.Change(self.variable) # # def Change(self, var): # var = 'Changed' """ self.variable是對字符串對象的引用'Original'。 當您調用時Change,創建var對該對象的第二個引用。 在函數內部,您將引用重新分配var給不同的字符串對象'Changed',但引用self.variable是獨立的,不會更改。 解決這個問題的唯一方法是傳遞一個可變對象。因為兩個引用都引用同一個對象,所以對象的任何更改都會反映在兩個位置。 """ # def __init__(self): # self.variable = ['Original'] # self.Change(self.variable) # # def Change(self, var): # var[0] = 'Changed' """ 對上面的評論: 它既不是按值傳遞,也不是按引用傳遞 - 它是逐個調用的。請見Fredrik Lundh: http://effbot.org/zone/call-by-object.htm 這是一個重要的引用: “......變量[名稱] 不是對象;它們不能用其他變量表示或由對象引用。” 在您的示例中,Change調用方法時- 為其創建名稱空間 ; 並var成為該命名空間中字符串對象的名稱'Original'。 然后,該對象在兩個名稱空間中具有名稱。接下來,var = 'Changed'綁定var到一個新的字符串對象,因此該方法的命名空間忘記了'Original'。 最后,忘記了該命名空間,並將字符串'Changed'與它一起使用。 """
# x = [ 2, 4, 4, 5, 5 ] # print (x) # 2, 4, 4, 5, 5 # # def go( li ) : # li = [ 5, 6, 7, 8 ] # re-assigning what li POINTS TO, does not # # change the value of the ORIGINAL variable x # print("go_li",li) # go( x ) # # print (x) # 2, 4, 4, 5, 5 [ STILL! ] # # # print( 'press any key to continue' ) """ 有趣的總結 事實是,整個參考/值概念將不適合python。Python沒有變量的“價值”。Python只有引用對象的對象和名稱。 因此,當您調用函數並在括號內放置“名稱”時,如下所示: def func(x): # defines a function that takes an argument ... # do something here func(myname) # calling the function myname傳遞指向的實際對象,而不是名稱 myname 本身。在函數內部,給出了另一個name(x)來引用傳遞的同一個對象。 您可以修改函數內部的對象(如果它是可變的),但您無法更改外部名稱指向的對象。就像你做的那樣 anothername = myname 因此,我可以回答你的問題: 它是“按值傳遞”,但所有值都只是對象的引用。 """ # a=1 # print("a",id(a)) # def nuw_1(data): # print("data",id(data)) # data+=1 # # print("data",id(data)) # return data # # a=nuw_1(a) # print("a",id(a)) a=[] print("a",id(a)) def nuw_1(data): print("data",data,id(data)) data=1 print("data",data,id(data)) nuw_1(a) print("a",a,id(a)) """
輸出
a 2072004795720
data [] 2072004795720
data 1 1604579120
a [] 2072004795720
""" a=[] print("a",id(a)) def nuw_1(data): print("data",data,id(data)) data.append(1) print("data",data,id(data)) nuw_1(a) print("a",a,id(a))
"""
輸出:
a 2072004793288
data [] 2072004793288
data [1] 2072004793288
a [1] 2072004793288
"""