1. 程式人生 > >python函式在傳參的時候,到底在傳些什麼?

python函式在傳參的時候,到底在傳些什麼?

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]