日韩久久久精品,亚洲精品久久久久久久久久久,亚洲欧美一区二区三区国产精品 ,一区二区福利

Python函數(shù)中參數(shù)是傳遞值還是引用詳解

系統(tǒng) 1704 0

在 C/C++ 中,傳值和傳引用是函數(shù)參數(shù)傳遞的兩種方式,在Python中參數(shù)是如何傳遞的?回答這個問題前,不如先來看兩段代碼。

代碼段1:

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

看了代碼段1的同學(xué)可能會說參數(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 中變量與對象之間的關(guān)系。

變量與對象

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

            
a = []
a.append(1)
          


在 Python 中,「變量」更準(zhǔn)確叫法是「名字」,賦值操作 = 就是把一個名字綁定到一個對象上。就像給對象添加一個標(biāo)簽。

a = 1

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

a = 2

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

b = a

把變量 a 賦值給另外一個變量 b,相當(dāng)于在對象 2 上貼了 a,b 兩個標(biāo)簽,通過這兩個變量都可以對對象 2 進(jìn)行操作。

變量本身沒有類型信息,類型信息存儲在對象中,這和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ù)中參數(shù)是傳遞值還是引用詳解_第1張圖片

在代碼段1中,變量 a 綁定了 1,調(diào)用函數(shù) foo(a) 時,相當(dāng)于給參數(shù) arg 賦值 arg=1,這時兩個變量都綁定了 1。在函數(shù)里面 arg 重新賦值為 2 之后,相當(dāng)于把 1 上的 arg 標(biāo)簽撕掉,貼到 2 身上,而 1 上的另外一個標(biāo)簽 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ù)中參數(shù)是傳遞值還是引用詳解_第2張圖片

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

最后,回到問題本身,究竟是是傳值還是傳引用呢?說傳值或者傳引用都不準(zhǔn)確。非要安一個確切的叫法的話,叫傳對象(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
          

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

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

Python函數(shù)中參數(shù)是傳遞值還是引用詳解_第3張圖片

而正確的方式是,把參數(shù)默認(rèn)值指定為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ù)中參數(shù)是傳遞值還是引用詳解_第4張圖片

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。


更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號聯(lián)系: 360901061

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對您有幫助就好】

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長會非常 感謝您的哦?。?!

發(fā)表我的評論
最新評論 總共0條評論
主站蜘蛛池模板: 江陵县| 青神县| 沂源县| 分宜县| 榆树市| 威海市| 邹城市| 平江县| 洪泽县| 凤冈县| 汉川市| 沧源| 巩留县| 苍梧县| 城市| 榕江县| 张家港市| 武胜县| 微博| 中西区| 邵阳市| 克拉玛依市| 吴桥县| 虞城县| 九龙县| 恭城| 房产| 阳新县| 乳山市| 霍林郭勒市| 灵川县| 江都市| 锡林浩特市| 义马市| 梨树县| 柞水县| 驻马店市| 霍城县| 沛县| 札达县| 邳州市|