python函式在傳參的時候,到底在傳些什麼?
阿新 • • 發佈:2019-02-16
C++這樣的語言用多了之後,在Python函式傳遞引數的時候,經常會遇到一個問題,我要傳遞一個引用怎麼辦?
比如我們想要傳一個x到函式中做個運算改變x的值:
def change(y):
y += 1
x = 1
print ("before change:", x)
change(x)
print ("after change: ", x)
得到的結果是
before change: 1
after change: 1
完全沒用~~~這是怎麼回事?
我來說明一下這個過程~
也就是說python中的等號就是把變數指向了一個例項而已。
那麼,如何做到其它語言中引用這樣的效果呢?一個比較推薦的方法就是用把引數return回去就可以了,如下面程式碼所示。
def change(y):
y += 1
return y
x = 1
print ("before change:", x)
x = change(x)
print ("after change: ", x)
結果為
before change: 1
after change: 2
另一種方法就是,我們不讓函式中的引數指向新的地址不就可以了,我們讓它在原來的地址上修改我們的值。這一點,只有一部分的資料型別可以做到,我們把這樣的物件叫做可變物件,做不到的就叫不可變物件。
- 不可變物件:int,string,float,tuple 等
- 可變物件 :list,dictionary 等
舉幾個例子更為通俗易懂
def change(x):
print ("before append:", x, "id:", id(x))
x.append(0)
print ("after append:", x, "id:", id(x))
x = [1, 2, 3]
print ("before change:", x, "id:", id(x))
change(x)
print ("after change:", x, "id:", id(x))
before change: [1, 2, 3] id: 140623459512008
before append: [1 , 2, 3] id: 140623459512008
after append: [1, 2, 3, 0] id: 140623459512008
after change: [1, 2, 3, 0] id: 140623459512008
由於list在做append操作時,x指向的地址是不變的,所以可以有引用的效果,x[0] = 4這樣的操作也是沒問題的
然而,如果是把x指向了新的地址就不行了,如下
def change(x):
print ("before equal:", x, "id:", id(x))
x = [5]
print ("after equal:", x, "id:", id(x))
x = [1, 2, 3]
print ("before change:", x, "id:", id(x))
change(x)
print ("after change:", x, "id:", id(x))
before change: [1, 2, 3] id: 139683058095432
before equal: [1, 2, 3] id: 139683058095432
after equal: [5] id: 139683058028360
after change: [1, 2, 3] id: 139683058095432
這下應該就搞清楚是怎麼回事了吧,不過正是因為python的這種特性,在函式設定預設引數的時候,我們要注意儘量不用可變物件當預設值,否則會發生下面這種情況
def change(x=[]):
x.append(1)
return x
print (change())
print (change())
[1]
[1, 1]
比較妥當的做法是用不可變物件來代替[],比如None
def change(x=None):
if not x:
x = []
x.append(1)
return x
print (change())
print (change())
此時的輸出為
[1]
[1]