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


做者:winterTTr (轉載請註明)python

我想,這個標題或許是不少初學者的問題。尤爲是像我這樣的對C/C++比較熟悉,剛剛進入python殿堂的朋友們函數

。C/C++的函數參數的傳遞方式根深蒂固的影響這咱們的思惟--引用?傳值?到底是那種呢。spa

呵呵,語言的特性決定了是使用的方法,那麼,如今咱們來探究一下python的函數參數傳遞方式。對象

在開始以前,咱們有必要分清一下python的一些基礎概念。內存

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

在python中,類型屬於對象,變量是沒有類型的,這正是python的語言特性,也是吸引着不少pythoner的一點。全部的變量均可以理解是內存中一個對象的「引用」,或者,也能夠看似c中void*的感受。因此,但願你們在看到一個python變量的時候,把變量和真正的內存對象分開。table

類型是屬於對象的,而不是變量。這樣,不少問題就容易思考了。基礎

例如: 變量

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

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中的變量和對象的概念有了更深刻的理解了呢?

經過我上面的解釋,我想你們也能夠本身搞定其餘類型對象的傳遞問題了吧。

相關文章
相關標籤/搜索