Python參數(shù)傳遞機(jī)制傳值和傳引用原理詳解
首先還是應(yīng)該科普下函數(shù)參數(shù)傳遞機(jī)制,傳值和傳引用是什么意思?
函數(shù)參數(shù)傳遞機(jī)制問(wèn)題在本質(zhì)上是調(diào)用函數(shù)(過(guò)程)和被調(diào)用函數(shù)(過(guò)程)在調(diào)用發(fā)生時(shí)進(jìn)行通信的方法問(wèn)題?;镜膮?shù)傳遞機(jī)制有兩種:值傳遞和引用傳遞。
值傳遞(passl-by-value)過(guò)程中,被調(diào)函數(shù)的形式參數(shù)作為被調(diào)函數(shù)的局部變量處理,即在堆棧中開(kāi)辟了內(nèi)存空間以存放由主調(diào)函數(shù)放進(jìn)來(lái)的實(shí)參的值,從而成為了實(shí)參的一個(gè)副本。值傳遞的特點(diǎn)是被調(diào)函數(shù)對(duì)形式參數(shù)的任何操作都是作為局部變量進(jìn)行,不會(huì)影響主調(diào)函數(shù)的實(shí)參變量的值。
引用傳遞(pass-by-reference)過(guò)程中,被調(diào)函數(shù)的形式參數(shù)雖然也作為局部變量在堆棧中開(kāi)辟了內(nèi)存空間,但是這時(shí)存放的是由主調(diào)函數(shù)放進(jìn)來(lái)的實(shí)參變量的地址。被調(diào)函數(shù)對(duì)形參的任何操作都被處理成間接尋址,即通過(guò)堆棧中存放的地址訪問(wèn)主調(diào)函數(shù)中的實(shí)參變量。正因?yàn)槿绱?,被調(diào)函數(shù)對(duì)形參做的任何操作都影響了主調(diào)函數(shù)中的實(shí)參變量。
在python中實(shí)際又是怎么樣的呢?
先看一個(gè)簡(jiǎn)單的例子
from ctypes import *
import os.path
import sys
def test(c):
print "test before "
print id(c)
c+=2
print "test after +"
print id(c)
return c
def printIt(t):
for i in range(len(t)):
print t[i]
if __name__=="__main__":
a=2
print "main before invoke test"
print id(a)
n=test(a)
print "main afterf invoke test"
print a
print id(a)
運(yùn)行后結(jié)果如下:
>>> main before invoke test test before test after + main afterf invoke test 39601564
id函數(shù)可以獲得對(duì)象的內(nèi)存地址.很明顯從上面例子可以看出,將a變量作為參數(shù)傳遞給了test函數(shù),傳遞了a的一個(gè)引用,把a(bǔ)的地址傳遞過(guò)去了,所以在函數(shù)內(nèi)獲取的變量C的地址跟變量a的地址是一樣的,但是在函數(shù)內(nèi),對(duì)C進(jìn)行賦值運(yùn)算,C的值從2變成了4,實(shí)際上2和4所占的內(nèi)存空間都還是存在的,賦值運(yùn)算后,C指向4所在的內(nèi)存。而a仍然指向2所在的內(nèi)存,所以后面打印a,其值還是2.
如果還不能理解,先看下面例子
>>> a=1 >>> b=1 >>> id(a) >>> id(b) >>> a=2 >>> id(a)
a和b都是int類(lèi)型的值,值都是1,而且內(nèi)存地址都是一樣的,這已經(jīng)表明了在python中,可以有多個(gè)引用指向同一個(gè)內(nèi)存(畫(huà)了一個(gè)很挫的圖,見(jiàn)諒),在給a賦值為2后,再次查看a的內(nèi)存地址,都已經(jīng)變化了

而基于最前面的例子,大概可以這樣描述:

那python函數(shù)傳參就是傳引用?然后傳參的值在被調(diào)函數(shù)內(nèi)被修改也不影響主調(diào)函數(shù)的實(shí)參變量的值?再來(lái)看個(gè)例子。
from ctypes import *
import os.path
import sys
def test(list2):
print "test before "
print id(list2)
list2[1]=30
print "test after +"
print id(list2)
return list2
def printIt(t):
for i in range(len(t)):
print t[i]
if __name__=="__main__":
list1=["loleina",25,'female']
print "main before invoke test"
print id(list1)
list3=test(list1)
print "main afterf invoke test"
print list1
print id(list1)
實(shí)際值為:
>>> main before invoke test test before test after + main afterf invoke test ['loleina', 30, 'female']
發(fā)現(xiàn)一樣的傳值,而第二個(gè)變量居然變化,為啥呢?
實(shí)際上是因?yàn)閜ython中的序列:列表是一個(gè)可變的對(duì)象,就基于list1=[1,2] list1[0]=[0]這樣前后的查看list1的內(nèi)存地址,是一樣的。
>>> list1=[1,2] >>> id(list1) >>> list1[0]=[0] >>> list1 [[0], 2] >>> id(list1)
字典也是可變對(duì)象:
>>> def fun2(num1,l1,d1):
... num1=123
... l1[0]=123
... d1['a']=123
... print("inside:","num1=%f,l1=%s,d1=%s"%(num1,l1,d1))
...
>>> num=111
>>> l=[1,1,1]
>>> d={'a':111,'b':0}
>>> print("before:","num=%f,l=%s,d=%s"%(num,l,d))
before: num=111.000000,l=[1, 1, 1],d={'a': 111, 'b': 0}
>>> fun2(num,l,d)
inside: num1=123.000000,l1=[123, 1, 1],d1={'a': 123, 'b': 0}
>>> print("after:","num=%f,l=%s,d=%s"%(num,l,d))
after: num=111.000000,l=[123, 1, 1],d={'a': 123, 'b': 0}
結(jié)論:python不允許程序員選擇采用傳值還是傳引用。Python參數(shù)傳遞采用的肯定是“傳對(duì)象引用”的方式。這種方式相當(dāng)于傳值和傳引用的一種綜合。
如果函數(shù)收到的是一個(gè)可變對(duì)象(比如字典或者列表)的引用,就能修改對(duì)象的原始值--相當(dāng)于通過(guò)“傳引用”來(lái)傳遞對(duì)象。
如果函數(shù)收到的是一個(gè)不可變對(duì)象(比如數(shù)字、字符或者元組)的引用,就不能直接修改原始對(duì)象--相當(dāng)于通過(guò)“傳值'來(lái)傳遞對(duì)象。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
使用python對(duì)pdf文件進(jìn)行加密等操作
這篇文章主要為大家詳細(xì)介紹了使用python對(duì)pdf文件進(jìn)行加密等操作的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-12-12
Python用K-means聚類(lèi)算法進(jìn)行客戶分群的實(shí)現(xiàn)
這篇文章主要介紹了Python用K-means聚類(lèi)算法進(jìn)行客戶分群的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08
一文詳解PyQt5中信號(hào)(Signal)與槽(Slot)
信號(hào)(Signal)與槽(Slot)是Qt中的核心機(jī)制,也是在PyQt編程中對(duì)象之間進(jìn)行通信的機(jī)制。這篇文章主要帶大家了解一下信號(hào)(Signal)與槽(Slot)的使用,需要的可以參考一下2022-12-12
Python實(shí)現(xiàn)為圖片添加水印的示例詳解
這篇文章主要介紹了如何通過(guò)Python3實(shí)現(xiàn)添加水印,這樣發(fā)朋友圈,圖片再也不怕被盜了?。?!文中的示例代碼簡(jiǎn)潔易懂,需要的可以參考一下2022-02-02
基于PyQt5實(shí)現(xiàn)一個(gè)串口接數(shù)據(jù)波形顯示工具
這篇文章主要為大家詳細(xì)介紹了如何利用PyQt5實(shí)現(xiàn)一個(gè)串口接數(shù)據(jù)波形顯示工具,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起了解一下2023-01-01
python利用paramiko連接遠(yuǎn)程服務(wù)器執(zhí)行命令的方法
下面小編就為大家?guī)?lái)一篇python利用paramiko連接遠(yuǎn)程服務(wù)器執(zhí)行命令的方法。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-10-10
python json.dumps中文亂碼問(wèn)題解決
這篇文章主要介紹了python json.dumps中文亂碼問(wèn)題解決,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04
python 遞歸調(diào)用返回None的問(wèn)題及解決方法
這篇文章主要介紹了python 遞歸調(diào)用返回None的問(wèn)題,本文通過(guò)實(shí)例代碼給大家記錄了解決方案,代碼簡(jiǎn)單易懂,非常不錯(cuò)對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-03-03
Django admin 實(shí)現(xiàn)search_fields精確查詢(xún)實(shí)例
這篇文章主要介紹了Django admin 實(shí)現(xiàn)search_fields精確查詢(xún)實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-03-03

