Python如何通過引用傳遞變量?


"""
# 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

"""

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM