Python 函數中,參數是傳值,仍是傳引用?

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

 

代碼段1:面試

def foo(arg):

    arg = 2

    print(arg)

 

a = 1

foo(a)  # 輸出:2

print(a) # 輸出:1

看了代碼段1的同窗可能會說參數是值傳遞。app

 

代碼段2:函數

def bar(args):

    args.append(1)

 

b = []

print(b)# 輸出:[]

print(id(b)) # 輸出:4324106952

bar(b)

print(b) # 輸出:[1]

print(id(b))  # 輸出:4324106952

看了代碼段2,這時可能又有人會說,參數是傳引用,那麼問題來了,參數傳遞究竟是傳值仍是傳引用或者二者都不是?爲了把這個問題弄清楚,先了解 Python 中變量與對象之間的關係。spa

 

變量與對象code

 

Python 中一切皆爲對象,數字是對象,列表是對象,函數也是對象,任何東西都是對象。而變量是對象的一個引用(又稱爲名字或者標籤),對象的操做都是經過引用來完成的。例如,[]是一個空列表對象,變量 a 是該對象的一個引用對象

1
2
3
=  []
 
a.append( 1 )

在 Python 中,「變量」更準確叫法是「名字」,賦值操做 = 就是把一個名字綁定到一個對象上。就像給對象添加一個標籤。blog

1
=  1

                                          

 

整數 1 賦值給變量 a 就至關因而在整數1上綁定了一個 a 標籤。內存

1
=  2

                                          

 

整數 2 賦值給變量 a,至關於把原來整數 1 身上的 a 標籤撕掉,貼到整數 2 身上。ci

1
=  a

                                        

 

把變量 a 賦值給另一個變量 b,至關於在對象 2 上貼了 a,b 兩個標籤,經過這兩個變量均可以對對象 2 進行操做。

 

變量自己沒有類型信息,類型信息存儲在對象中,這和C/C++中的變量有很是大的出入(C中的變量是一段內存區域)

 

函數參數

 

Python 函數中,參數的傳遞本質上是一種賦值操做,而賦值操做是一種名字到對象的綁定過程,清楚了賦值和參數傳遞的本質以後,如今再來分析前面兩段代碼。

1
2
3
4
5
6
7
8
9
10
11
12
13
def  foo(arg):
 
     arg  =  2
 
     print (arg)
 
  
 
=  1
 
foo(a)   # 輸出:2
 
print (a)  # 輸出:1

                                        

 

在代碼段1中,變量 a 綁定了 1,調用函數 foo(a) 時,至關於給參數 arg 賦值 arg=1,這時兩個變量都綁定了 1。在函數裏面 arg 從新賦值爲 2 以後,至關於把 1 上的 arg 標籤撕掉,貼到 2 身上,而 1 上的另一個標籤 a 一直存在。所以 print(a) 仍是 1。

 

再來看一下代碼段2

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
def  bar(args):
 
     args.append( 1 )
 
  
 
=  []
 
print (b) # 輸出:[]
 
print ( id (b))  # 輸出:4324106952
 
bar(b)
 
print (b) # 輸出:[ 1 ]
 
print ( id (b))   # 輸出:4324106952

                                        

 

執行 append 方法前 b 和 arg 都指向(綁定)同一個對象,執行 append 方法時,並無從新賦值操做,也就沒有新的綁定過程,append 方法只是對列表對象插入一個元素,對象仍是那個對象,只是對象裏面的內容變了。由於 b 和 arg 都是綁定在同一個對象上,執行 b.append 或者 arg.append 方法本質上都是對同一個對象進行操做,所以 b 的內容在調用函數後發生了變化(但id沒有變,仍是原來那個對象)

 

最後,回到問題自己,到底是是傳值仍是傳引用呢?說傳值或者傳引用都不許確。非要安一個確切的叫法的話,叫傳對象(call by object)。若是做爲面試官,非要考察候選人對 Python 函數參數傳遞掌握與否,與其討論字面上的意思,還不如來點實際代碼。

 

show me the code

 

1
2
3
4
5
def  bad_append(new_item, a_list = []):
 
     a_list.append(new_item)
 
     return  a_list

 

該函數初衷是但願當沒有提供默認的a_list參數的時候,創建一個空的列表,而後把一個對象添加到該列表中,而且打印, 可是程序的執行彷佛並無按照咱們的意圖來執行,彷佛每一個函數都共享了a_list,可是咱們明明每次都在參數列表中初始化了a_list變量,分析以下:

python中的def語句在每次執行的時候都初始化一個函數對象,這個函數對象就是咱們要調用的函數,能夠把它當成一個通常的對象,只不過這個對象擁有一個可執行的方法和部分屬性,對於參數中提供了初始值的參數,因爲python中的函數參數傳遞的是對象,也能夠認爲是傳地址,只有def的時候初始化一次,而後在調用者和被調用者中都是共享的,因此在 func.func_defaults中只能看到一個默認參數,在該函數對象被初始化的時候就已經存在了。

 

1
2
3
4
5
6
7
>>>  print  bad_append( 'one' )
 
[ 'one' ]
 
>>>  print  bad_append( 'one' )
 
[ 'one' 'one' ]

 

                                   

 

而正確的方式是,把參數默認值指定爲None

1
2
3
4
5
6
7
8
9
def  good_append(new_item, a_list = None ):
 
     if  a_list  is  None :
 
         a_list  =  []
 
     a_list.append(new_item)
 
     return  a_list

  

                                    

相關文章
相關標籤/搜索