溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Python的參數(shù)是傳值,還是傳引用

發(fā)布時間:2022-01-17 15:35:55 來源:億速云 閱讀:152 作者:iii 欄目:大數(shù)據(jù)

本篇內(nèi)容介紹了“Python的參數(shù)是傳值,還是傳引用”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠?qū)W有所成!

在 C/C++ 中,傳值和傳引用是函數(shù)參數(shù)傳遞的兩種方式,學 Python 時,有人喜歡生搬硬套地問類似的問題:“Python 函數(shù)中,參數(shù)是傳值,還是傳引用?”。

回答這個問題前,不如先來看兩段代碼。

代碼段1:

def foo(arg):
   arg = 2
   print(arg)

a = 1
foo(a)  # 輸出:2
print(a) # 輸出:1

看了代碼段1的同學可能會說參數(shù)是值傳遞。

代碼段2:

def bar(args):
   args.append(1)

b = []
print(b)# 輸出:[]
print(id(b)) # 輸出:4324106952
bar(b)
print(b) # 輸出:[1]
print(id(b))  # 輸出:4324106952

看了代碼段2,這時可能又有人會說,參數(shù)是傳引用,那么問題來了,參數(shù)傳遞到底是傳值還是傳引用或者兩者都不是?

為了把這個問題弄清楚,先了解 Python 中變量與對象之間的關系。

變量與對象

Python 中一切皆為對象,數(shù)字是對象,列表是對象,函數(shù)也是對象,任何東西都是對象。而變量是對象的一個引用(又稱為名字或者標簽),對象的操作都是通過引用來完成的。例如,[]是一個空列表對象,變量 a 是該對象的一個引用

a = []
a.append(1)

在 Python 中,「變量」更準確叫法是「名字」,賦值操作 = 就是把一個名字綁定到一個對象上。就像給對象添加一個標簽。例如:

a = 1

Python的參數(shù)是傳值,還是傳引用

整數(shù) 1 賦值給變量 a 就相當于是在整數(shù)1上綁定了一個 a 標簽。

a = 2

Python的參數(shù)是傳值,還是傳引用

整數(shù) 2 賦值給變量 a,相當于把原來整數(shù) 1 身上的 a 標簽撕掉,貼到整數(shù) 2 身上。

b = a

Python的參數(shù)是傳值,還是傳引用

把變量 a 賦值給另外一個變量 b,相當于在對象 2 上貼了 a,b 兩個標簽,通過這兩個變量都可以對對象 2 進行操作。

變量本身沒有類型信息,類型信息存儲在對象中,這和C/C++中的變量有非常大的出入(C中的變量是一段內(nèi)存區(qū)域)

函數(shù)參數(shù)

Python 函數(shù)中,參數(shù)的傳遞本質(zhì)上是一種賦值操作,而賦值操作是一種名字到對象的綁定過程,清楚了賦值和參數(shù)傳遞的本質(zhì)之后,現(xiàn)在再來分析前面兩段代碼。

def foo(arg):
   arg = 2
   print(arg)

a = 1
foo(a)  # 輸出:2
print(a) # 輸出:1

Python的參數(shù)是傳值,還是傳引用

在代碼段1中,變量 a 綁定了 1,調(diào)用函數(shù) foo(a) 時,相當于給參數(shù) arg 賦值 arg=1,這時兩個變量都綁定了 1。在函數(shù)里面 arg 重新賦值為 2 之后,相當于把 1 上的 arg 標簽撕掉,貼到 2 身上,而 1 上的另外一個標簽 a 一直存在。因此 print(a) 還是 1。

再來看一下代碼段2

def bar(args):
   args.append(1)

b = []
print(b)# 輸出:[]
print(id(b)) # 輸出:4324106952
bar(b)
print(b) # 輸出:[1]
print(id(b))  # 輸出:4324106952

Python的參數(shù)是傳值,還是傳引用

執(zhí)行 append 方法前 b 和 arg 都指向(綁定)同一個對象,執(zhí)行 append 方法時,并沒有重新賦值操作,也就沒有新的綁定過程,append 方法只是對列表對象插入一個元素,對象還是那個對象,只是對象里面的內(nèi)容變了。因為 b 和 arg 都是綁定在同一個對象上,執(zhí)行 b.append 或者 arg.append 方法本質(zhì)上都是對同一個對象進行操作,因此 b 的內(nèi)容在調(diào)用函數(shù)后發(fā)生了變化(但id沒有變,還是原來那個對象)

最后,回到問題本身,究竟是是傳值還是傳引用呢?說傳值或者傳引用都不準確。非要安一個確切的叫法的話,叫傳對象(call by object)。如果作為面試官,非要考察候選人對 Python 函數(shù)參數(shù)傳遞掌握與否,與其討論字面上的意思,還不如來點實際代碼。

show me the code

def bad_append(new_item, a_list=[]):
    a_list.append(new_item)
    return a_list

這段代碼是初學者最容易犯的錯誤,用可變(mutable)對象作為參數(shù)的默認值。函數(shù)定義好之后,默認參數(shù) a_list 就會指向(綁定)到一個空列表對象,每次調(diào)用函數(shù)時,都是對同一個對象進行 append 操作。因此這樣寫就會有潛在的bug,同樣的調(diào)用方式返回了不一樣的結果。

>>> print bad_append('one')
['one']
>>> print bad_append('one')
['one', 'one']

Python的參數(shù)是傳值,還是傳引用

而正確的方式是,把參數(shù)默認值指定為None

def good_append(new_item, a_list=None):
    if a_list is None:
        a_list = []
    a_list.append(new_item)
    return a_list

Python的參數(shù)是傳值,還是傳引用

“Python的參數(shù)是傳值,還是傳引用”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關的知識可以關注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實用文章!

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權內(nèi)容。

AI