作者: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中的變數和對象的概念有了更深入的理解了呢?
通過我上面的解釋,我想大家也可以自己搞定其他類型對象的傳遞問題了吧。