Python函數(shù)中參數(shù)是傳遞值還是引用詳解
在 C/C++ 中,傳值和傳引用是函數(shù)參數(shù)傳遞的兩種方式,在Python中參數(shù)是如何傳遞的?回答這個(gè)問題前,不如先來看兩段代碼。
代碼段1:
def foo(arg): arg = 2 print(arg) a = 1 foo(a) # 輸出:2 print(a) # 輸出:1
看了代碼段1的同學(xué)可能會(huì)說參數(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í)可能又有人會(huì)說,參數(shù)是傳引用,那么問題來了,參數(shù)傳遞到底是傳值還是傳引用或者兩者都不是?為了把這個(gè)問題弄清楚,先了解 Python 中變量與對(duì)象之間的關(guān)系。
變量與對(duì)象
Python 中一切皆為對(duì)象,數(shù)字是對(duì)象,列表是對(duì)象,函數(shù)也是對(duì)象,任何東西都是對(duì)象。而變量是對(duì)象的一個(gè)引用(又稱為名字或者標(biāo)簽),對(duì)象的操作都是通過引用來完成的。例如,[]是一個(gè)空列表對(duì)象,變量 a 是該對(duì)象的一個(gè)引用
a = [] a.append(1)
在 Python 中,「變量」更準(zhǔn)確叫法是「名字」,賦值操作 = 就是把一個(gè)名字綁定到一個(gè)對(duì)象上。就像給對(duì)象添加一個(gè)標(biāo)簽。
a = 1
整數(shù) 1 賦值給變量 a 就相當(dāng)于是在整數(shù)1上綁定了一個(gè) a 標(biāo)簽。
a = 2
整數(shù) 2 賦值給變量 a,相當(dāng)于把原來整數(shù) 1 身上的 a 標(biāo)簽撕掉,貼到整數(shù) 2 身上。
b = a
把變量 a 賦值給另外一個(gè)變量 b,相當(dāng)于在對(duì)象 2 上貼了 a,b 兩個(gè)標(biāo)簽,通過這兩個(gè)變量都可以對(duì)對(duì)象 2 進(jìn)行操作。
變量本身沒有類型信息,類型信息存儲(chǔ)在對(duì)象中,這和C/C++中的變量有非常大的出入(C中的變量是一段內(nèi)存區(qū)域)
函數(shù)參數(shù)
Python 函數(shù)中,參數(shù)的傳遞本質(zhì)上是一種賦值操作,而賦值操作是一種名字到對(duì)象的綁定過程,清楚了賦值和參數(shù)傳遞的本質(zhì)之后,現(xiàn)在再來分析前面兩段代碼。
def foo(arg): arg = 2 print(arg) a = 1 foo(a) # 輸出:2 print(a) # 輸出:1
在代碼段1中,變量 a 綁定了 1,調(diào)用函數(shù) foo(a) 時(shí),相當(dāng)于給參數(shù) arg 賦值 arg=1,這時(shí)兩個(gè)變量都綁定了 1。在函數(shù)里面 arg 重新賦值為 2 之后,相當(dāng)于把 1 上的 arg 標(biāo)簽撕掉,貼到 2 身上,而 1 上的另外一個(gè)標(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
執(zhí)行 append 方法前 b 和 arg 都指向(綁定)同一個(gè)對(duì)象,執(zhí)行 append 方法時(shí),并沒有重新賦值操作,也就沒有新的綁定過程,append 方法只是對(duì)列表對(duì)象插入一個(gè)元素,對(duì)象還是那個(gè)對(duì)象,只是對(duì)象里面的內(nèi)容變了。因?yàn)?b 和 arg 都是綁定在同一個(gè)對(duì)象上,執(zhí)行 b.append 或者 arg.append 方法本質(zhì)上都是對(duì)同一個(gè)對(duì)象進(jìn)行操作,因此 b 的內(nèi)容在調(diào)用函數(shù)后發(fā)生了變化(但id沒有變,還是原來那個(gè)對(duì)象)
最后,回到問題本身,究竟是是傳值還是傳引用呢?說傳值或者傳引用都不準(zhǔn)確。非要安一個(gè)確切的叫法的話,叫傳對(duì)象(call by object)。如果作為面試官,非要考察候選人對(duì) Python 函數(shù)參數(shù)傳遞掌握與否,與其討論字面上的意思,還不如來點(diǎn)實(shí)際代碼。
show me the code
def bad_append(new_item, a_list=[]): a_list.append(new_item) return a_list
這段代碼是初學(xué)者最容易犯的錯(cuò)誤,用可變(mutable)對(duì)象作為參數(shù)的默認(rèn)值。函數(shù)定義好之后,默認(rèn)參數(shù) a_list 就會(huì)指向(綁定)到一個(gè)空列表對(duì)象,每次調(diào)用函數(shù)時(shí),都是對(duì)同一個(gè)對(duì)象進(jìn)行 append 操作。因此這樣寫就會(huì)有潛在的bug,同樣的調(diào)用方式返回了不一樣的結(jié)果。
>>> print bad_append('one') ['one'] >>> print bad_append('one') ['one', 'one']
而正確的方式是,把參數(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
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
VSCode配置python環(huán)境及中文問題解決方法
這篇文章主要介紹了VSCode配置python環(huán)境及中文問題,print打印中文亂碼如何解決這個(gè)問題呢,本文給大家?guī)韮煞N方法幫助大家解決這個(gè)問題,需要的朋友可以參考下2022-02-02python進(jìn)程管理工具supervisor的安裝與使用教程
supervisor是用python寫的一個(gè)進(jìn)程管理工具,用來啟動(dòng),重啟,關(guān)閉進(jìn)程。下面這篇文章主要給大家介紹了關(guān)于python實(shí)現(xiàn)的進(jìn)程管理工具supervisor的安裝與使用的相關(guān)資料,需要的朋友可以參考借鑒,下面來一起看看吧。2017-09-09Python2.7 實(shí)現(xiàn)引入自己寫的類方法
下面小編就為大家分享一篇Python2.7 實(shí)現(xiàn)引入自己寫的類方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-04-04Python爬蟲報(bào)錯(cuò)<response [406]>(已解決)
本文主要介紹了Python爬蟲報(bào)錯(cuò)<response [406]>,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-02-02Python capitalize()函數(shù)的用法詳解
在Python中,capitalize()將字符串的第一個(gè)字符轉(zhuǎn)換為大寫字母,并將所有其他字符(如果有的話)轉(zhuǎn)換為小寫,本文就將給大家介紹一下Python capitalize()函數(shù)的使用方法,感興趣的朋友跟著小編一起來看看吧2023-07-07Python解析JSON數(shù)據(jù)的基本方法實(shí)例代碼
JSON (JavaScript Object Notation) 是一種輕量級(jí)的數(shù)據(jù)交換格式,下面這篇文章主要給大家介紹了關(guān)于Python解析JSON數(shù)據(jù)的基本方法,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-01-01python中import和from-import的區(qū)別解析
這篇文章主要介紹了python中import和from-import的區(qū)別解析,本文通過實(shí)例代碼給大家講解的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-12-12python-pyinstaller、打包后獲取路徑的實(shí)例
今天小編就為大家分享一篇python-pyinstaller、打包后獲取路徑的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2019-06-06