欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

python 函數(shù)傳參之傳值還是傳引用的分析

 更新時(shí)間:2017年09月07日 17:31:02   作者:loleina  
這篇文章主要介紹了python 函數(shù)傳參之傳值還是傳引用的分析,需要的朋友可以參考下

首先還是應(yīng)該科普下函數(shù)參數(shù)傳遞機(jī)制,傳值和傳引用是什么意思?

   函數(shù)參數(shù)傳遞機(jī)制問題在本質(zhì)上是調(diào)用函數(shù)(過程)和被調(diào)用函數(shù)(過程)在調(diào)用發(fā)生時(shí)進(jìn)行通信的方法問題?;镜膮?shù)傳遞機(jī)制有兩種:值傳遞和引用傳遞。

  值傳遞(passl-by-value)過程中,被調(diào)函數(shù)的形式參數(shù)作為被調(diào)函數(shù)的局部變量處理,即在堆棧中開辟了內(nèi)存空間以存放由主調(diào)函數(shù)放進(jìn)來的實(shí)參的值,從而成為了實(shí)參的一個(gè)副本。值傳遞的特點(diǎn)是被調(diào)函數(shù)對(duì)形式參數(shù)的任何操作都是作為局部變量進(jìn)行,不會(huì)影響主調(diào)函數(shù)的實(shí)參變量的值。

  引用傳遞(pass-by-reference)過程中,被調(diào)函數(shù)的形式參數(shù)雖然也作為局部變量在堆棧中開辟了內(nèi)存空間,但是這時(shí)存放的是由主調(diào)函數(shù)放進(jìn)來的實(shí)參變量的地址。被調(diào)函數(shù)對(duì)形參的任何操作都被處理成間接尋址,即通過堆棧中存放的地址訪問主調(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

d函數(shù)可以獲得對(duì)象的內(nèi)存地址.很明顯從上面例子可以看出,將a變量作為參數(shù)傳遞給了test函數(shù),傳遞了a的一個(gè)引用,把a(bǔ)的地址傳遞過去了,所以在函數(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)

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

那python函數(shù)傳參就是傳引用?然后傳參的值在被調(diào)函數(shù)內(nèi)被修改也不影響主調(diào)函數(shù)的實(shí)參變量的值?再來看個(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)

結(jié)論:python不允許程序員選擇采用傳值還是傳引用。Python參數(shù)傳遞采用的肯定是“傳對(duì)象引用”的方式。這種方式相當(dāng)于傳值和傳引用的一種綜合。如果函數(shù)收到的是一個(gè)可變對(duì)象(比如字典或者列表)的引用,就能修改對(duì)象的原始值--相當(dāng)于通過“傳引用”來傳遞對(duì)象。如果函數(shù)收到的是一個(gè)不可變對(duì)象(比如數(shù)字、字符或者元組)的引用,就不能直接修改原始對(duì)象--相當(dāng)于通過“傳值'來傳遞對(duì)象。

相關(guān)文章

最新評(píng)論