歡迎來到Linux教程網
Linux教程網
Linux教程網
Linux教程網
Linux教程網 >> Linux編程 >> Linux編程 >> Python的函數參數傳遞:傳值?引用?

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

日期:2017/3/1 10:05:22   编辑:Linux編程

我想,這個標題或許是很多初學者的問題。尤其是像我這樣的對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中的變量和對象的概念有了更深入的理解了呢?

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

Copyright © Linux教程網 All Rights Reserved