深入理解python中函數(shù)傳遞參數(shù)是值傳遞還是引用傳遞
目前網(wǎng)絡(luò)上大部分博客的結(jié)論都是這樣的:
Python不允許程序員選擇采用傳值還是傳 引用。Python參數(shù)傳遞采用的肯定是“傳對(duì)象引用”的方式。實(shí)際上,這種方式相當(dāng)于傳值和傳引用的一種綜合。如果函數(shù)收到的是一個(gè)可變對(duì)象(比如字典 或者列表)的引用,就能修改對(duì)象的原始值——相當(dāng)于通過“傳引用”來傳遞對(duì)象。如果函數(shù)收到的是一個(gè)不可變對(duì)象(比如數(shù)字、字符或者元組)的引用,就不能 直接修改原始對(duì)象——相當(dāng)于通過“傳值”來傳遞對(duì)象。
你可以在很多討論該問題的博客里找到以上這一段話。
但是在實(shí)際操作中我卻發(fā)現(xiàn)一個(gè)問題:
l=[1,2,3] def a(x): x=x+[4] a(l) print(l)
這段代碼的輸出為:
[1,2,3]
為什么是這樣呢,list是可變對(duì)象,按照上面的結(jié)論來說傳遞方式是引用傳遞,我應(yīng)該在函數(shù)里能對(duì)它進(jìn)行修改呀?難道不應(yīng)該輸出[1,2,3,4]嗎?
我覺得我上面引用的那段大多數(shù)博主的結(jié)論,其實(shí)非常不好理解,而且沒有講到本質(zhì),看的云里霧里的。
經(jīng)過我后面的多次試驗(yàn),得到以下結(jié)論:
其實(shí)在python中討論值傳遞還是引用傳遞是沒有意義的,要真正對(duì)這些情況作出解釋,其實(shí)是應(yīng)該搞明白python(對(duì)可變對(duì)象和不可變對(duì)象的)賦值過程中是如何分配內(nèi)存地址的。
接下來,我們不討論值傳遞和引用傳遞的問題。
讓我們做一個(gè)非常簡單的小實(shí)驗(yàn),其中,id()可以查看變量在內(nèi)存中的地址:
l1=[1,2,3] l2=[1,2,3] a=1 b=1 print(id(l1)) print(id(l2)) print(id(a)) print(id(b))
在我的電腦中的運(yùn)行結(jié)果:
12856594504 12856915080 1643643344 1643643344
可以發(fā)現(xiàn),對(duì)于可變對(duì)象list來說,即便列表內(nèi)容一模一樣,python也會(huì)給它們分配新的不同的地址。
然而,對(duì)于不可變對(duì)象int來說,內(nèi)存里只有一個(gè)1。即便再定義一個(gè)變量c=1,也是指向內(nèi)存中同一個(gè)1。換句話說,不可變對(duì)象1的地址是共享的。
接下來讓我們看看在函數(shù)中調(diào)用可變對(duì)象和不可變對(duì)象,并修改他們的值,會(huì)是一個(gè)什么情況。
對(duì)于不可變對(duì)象int,我們來看看最簡單的情況:
a=1 print(id(a)) def x(a): print(id(a)) b=a print(id(b)) x(a)
運(yùn)行得到:
1643643344 1643643344 1643643344
這看起來就是一個(gè)引用傳遞,函數(shù)外的a、函數(shù)里的a和b都指向了同一個(gè)地址。
但我們?cè)賮砜匆粋€(gè)極端情況:
a=1 print(id(a)) def x(): b=1 print(id(b)) x()
運(yùn)行得到:
1643643344 1643643344
很神奇不是嗎?函數(shù)外定義的a和函數(shù)內(nèi)定義的b沒有任何關(guān)系,但它們指向同一個(gè)地址!
所以你說如何判斷它是值傳遞還是引用傳遞?討論這個(gè)問題根本沒有意義,因?yàn)閮?nèi)存里只有一個(gè)1。當(dāng)我把值1傳遞給函數(shù)里的某一個(gè)變量的時(shí)候,我實(shí)際上也傳遞了地址,因?yàn)閮?nèi)存里只有一個(gè)1。
甚至于說我直接給函數(shù)里的b賦值1都可以讓函數(shù)外的a和函數(shù)內(nèi)的b指向同一個(gè)地址。
下面來看看傳遞可變對(duì)象list的情況:
l=[1,2,3] print(id(l)) def a(x): print(id(x)) x.pop() print(x) print(id(x)) x=x+[3] print(x) print(id(x)) a(l)
運(yùn)行得到
883142451528 [1, 2] [1, 2, 3]
可以看到,當(dāng)我們把函數(shù)外的列表L傳遞給函數(shù)后,x的地址和L是一樣的,這看起來就是一個(gè)引用傳遞,沒問題。
繼續(xù)往下,我們調(diào)用x本身的方法pop后,x變成[1,2],并且x的地址沒變,這也沒什么問題。
但是當(dāng)我們給x賦值以后,x的地址就變了。
也就是說,只要?jiǎng)?chuàng)建一個(gè)新的可變對(duì)象,python就會(huì)分配一個(gè)新的地址。就算我們創(chuàng)建的新可變對(duì)象和已存在的舊可變對(duì)象完全一樣,python依舊會(huì)分配一個(gè)新的地址(見本文上半部分那個(gè)‘非常簡單的小實(shí)驗(yàn)')
而pop并不是創(chuàng)建新的可變對(duì)象,pop是對(duì)已有的可變對(duì)象進(jìn)行修改。
所以可以總結(jié)為:
在python中,不可變對(duì)象是共享的,創(chuàng)建可變對(duì)象永遠(yuǎn)是分配新地址
這個(gè)時(shí)候我們?cè)倩剡^頭來思考值傳遞和引用傳遞的問題,就會(huì)發(fā)現(xiàn)在python里討論這個(gè)確實(shí)是沒有意義。
我們可以說:python有著自己的一套特殊的傳參方式,這是由python動(dòng)態(tài)語言的性質(zhì)所決定的
總結(jié)
以上就是本文關(guān)于深入理解python中函數(shù)傳遞參數(shù)是值傳遞還是引用傳遞的全部內(nèi)容,希望對(duì)大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站:Python實(shí)現(xiàn)一個(gè)簡單的驗(yàn)證碼程序、Python編程django實(shí)現(xiàn)同一個(gè)ip十分鐘內(nèi)只能注冊(cè)一次、簡單了解Python中的幾種函數(shù)等,有什么問題可以隨時(shí)留言,小編會(huì)及時(shí)回復(fù)大家的。感謝朋友們對(duì)本站的支持!
相關(guān)文章
Flask利用自定義接口實(shí)現(xiàn)mock應(yīng)用詳解
后端接口已提供,前端需要依賴后端接口返回的數(shù)據(jù)進(jìn)行前端頁面的開發(fā),如何配合前端?這篇就來介紹一下Flask如何利用自定義接口實(shí)現(xiàn)mock應(yīng)用,需要的可以參考一下2023-03-03Python?調(diào)用GPT-3?API實(shí)現(xiàn)過程詳解
這篇文章主要為大家介紹了Python?調(diào)用GPT-3?API實(shí)現(xiàn)過程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-02-02Python for循環(huán)通過序列索引迭代過程解析
這篇文章主要介紹了Python for循環(huán)通過序列索引迭代過程解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-02-02python3 googletrans超時(shí)報(bào)錯(cuò)問題及翻譯工具優(yōu)化方案 附源碼
這篇文章主要介紹了python3 googletrans超時(shí)報(bào)錯(cuò)問題及翻譯工具優(yōu)化方案 附源碼,本文給大家分享解決方法,通過實(shí)例代碼相結(jié)合給大家介紹的非常詳細(xì),需要的朋友可以參考下2020-12-12Django中自定義admin Xadmin的實(shí)現(xiàn)代碼
這篇文章主要介紹了Django中自定義admin---Xadmin的實(shí)現(xiàn)代碼,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值 ,需要的朋友可以參考下2019-08-08Python基于Socket實(shí)現(xiàn)的簡單聊天程序示例
這篇文章主要介紹了Python基于Socket實(shí)現(xiàn)的簡單聊天程序,結(jié)合簡單實(shí)例形式分析了Python聊天程序的客戶端與服務(wù)器端相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2017-08-08python實(shí)現(xiàn)登錄密碼重置簡易操作代碼
這篇文章主要介紹了python實(shí)現(xiàn)登錄密碼重置簡易操作,代碼簡單易懂,非常不錯(cuò),具有一定的參考借鑒價(jià)值 ,需要的朋友可以參考下2019-08-08python中學(xué)習(xí)K-Means和圖片壓縮
大家在python中會(huì)遇到關(guān)于K-Means和圖片壓縮的問題,我先通過本次文章學(xué)習(xí)一下基本原理吧。2017-11-11