天天看點

Python的函數參數傳遞:傳值?引用?

作者:winterTTr (轉載請注明)

我想,這個标題或許是很多初學者的問題。尤其是像我這樣的對C/C++比較熟悉,剛剛進入python殿堂的朋友們

。C/C++的函數參數的傳遞方式根深蒂固的影響這我們的思維--引用?傳值?究竟是那種呢。

呵呵,語言的特性決定了是使用的方法,那麼,現在我們來探究一下python的函數參數傳遞方式。

在開始之前,我們有必要厘清一下python的一些基礎概念。

首先要說的是:變量 與 對象

在python中,類型屬于對象,變量是沒有類型的,這正是python的語言特性,也是吸引着很多pythoner的一點。所有的變量都可以了解是記憶體中一個對象的“引用”,或者,也可以看似c中void*的感覺。是以,希望大家在看到一個python變量的時候,把變量和真正的記憶體對象分開。

類型是屬于對象的,而不是變量。這樣,很多問題就容易思考了。

例如: 

nfoo = 1   #一個指向int資料類型的nfoo(再次提醒,nfoo沒有類型)

lstFoo = [1]   #一個指向list類型的lstFoo,這個list中包含一個整數1。

對應于上一個概念,就必須引出另了另一概念,這就是“可更改”(mutable)與“不可更改”(immutable)對象。

對于python比較熟悉的人們都應該了解這個事實,在python中,strings, tuples, 和numbers是不可更改的對象,而list,dict等則是可以修改的對象。那麼,這些所謂的可改變和不可改變影響着什麼呢?

還是上面的例子:

nfoo = 2        

這時,記憶體中原始的1對象因為不能改變,于是被“抛棄”,另nfoo指向一個新的int對象,其值為2

lstFoo[0] = 2  

更改list中第一個元素的值,因為list是可改變的,是以,第一個元素變更為2,其實應該說有一個新int對象被指定給lstFoo 所指向的對象的第一個值,但是對于lstFoo 來說,所指向的對象,并沒有變化,就是這個看似void*的變量所指向的對象仍舊是剛剛的那個有一個int對象的list。(聽着有點暈吧,仔細琢磨一下就明白了,嘿)

好了,被我這麼填鴨似的複習了一下python的基礎知識,改轉回題目的問題了,Python的函數參數傳遞:傳值?引用?

對于變量(與對象相對的概念),其實,python函數參數傳遞可以了解為就是變量傳值操作(注意哦,我說的是變量,不是對象  =_= )

接着說例子好了:

def ChangeInt( a ):

      a = 10  # change the number

nfoo = 2 

ChangeInt(nfoo)

print nfoo #結果是2

這時發生了什麼,有一個int對象2,和指向它的變量nfoo,當傳遞給ChangeInt的時候,按照傳值的方式,複制了變量nfoo的值,這樣,a就是nfoo指向同一個Int對象了,函數中a=10的時候,發生什麼?

(還記得我上面講到的那些概念麼),int是不能更改的對象,于是,做了一個新的int對象,另a指向它(但是此時,被變量nfoo指向的對象,沒有發生變化),于是在外面的感覺就是函數沒有改變nfoo的值,看起來像C++中的傳值方式。

def ChangeList( a ):

      a[0] = 10  # change the number

lstFoo = [2]

ChangeList(lstFoo )

print nfoo #結果是[10]

當傳遞給ChangeList的時候,變量仍舊按照“傳值”的方式,複制了變量lstFoo 的值,于是a和lstFoo 指向同一個對象,但是,list是可以改變的對象,對a[0]的操作,就是對lstFoo指向的對象的内容的操作,于是,這時的a[0] = 10,就是更改了lstFoo 指向的對象的第一個元素,是以,再次輸出lstFoo 時,顯示[10],内容被改變了,看起來,像C++中的按引用傳遞。

恩,現在是不是對python中的變量和對象的概念有了更深入的了解了呢?

通過我上面的解釋,我想大家也可以自己搞定其他類型對象的傳遞問題了吧。