python不變物件與可變物件
python變數賦值(可變與不可變)
知識點:python中,萬物皆物件。
python中不存在所謂的傳值呼叫,一切傳遞的都是物件的引用,也可以認為是傳址。
Python在heap中分配的物件分成兩類:可變物件和不可變物件。所謂可變物件是指,物件的內容是可變的,例如list。而不可變的物件則相反,表示其內容不可變。
- 不可變物件:int,string,float,tuple
- 可變物件 :list,dictionary
一、不可變物件
由於Python中的變數存放的是物件引用,所以對於不可變物件而言,儘管物件本身不可變,但變數的物件引用是可變的。運用這樣的機制,有時候會讓人產生糊塗,似乎可變物件變化了。如下面的程式碼:
i=73
i=i+2
不可變物件不變,引用變了
不可變的物件的特徵沒有變,依然是不可變物件,變的只是建立了新物件,改變了變數的物件引用。
看看下面的程式碼,更能體現這點的。
print(id(100)) # 1362346144
a = 100
print(id(a)) # 1362346144
b = 100
print(id(b)) # 1362346144
但如果變成258則id就有變化
print(id(258)) # 1659053321968 a = 258 print(id(a)) # 1659053321968 b = 258 print(id(b)) # 1659053319280
為何不一樣了?
二、可變物件
其物件的內容是可以變化的。當物件的內容發生變化時,變數的物件引用是不會變化的。
關於此問題的大坑:函式預設引數的使用
先定義一個函式,傳入一個list,新增一個END再返回:
def add_end(L=[]):
L.append('END')
return L
當你正常呼叫時,結果似乎不錯:
>>> add_end([1, 2, 3])
[1, 2, 3, 'END']
>>> add_end(['x', 'y', 'z'])
['x', 'y', 'z', 'END']
當你使用預設引數呼叫時,一開始結果也是對的:
>>> add_end()
['END']
但是,再次呼叫add_end()時,結果就不對了:
>>> add_end()
['END', 'END']
>>> add_end()
['END', 'END', 'END']
很多初學者很疑惑,預設引數是[],但是函式似乎每次都“記住了”上次添加了'END'後的list。
原因解釋如下:
Python函式在定義的時候,預設引數L的值就被計算出來了,即[],因為預設引數L也是一個變數,它指向物件[],每次呼叫該函式,如果改變了L的內容,則下次呼叫時,預設引數的內容就變了,不再是函式定義時的[]了。
所以,定義預設引數要牢記一點:預設引數必須指向不變物件!
要修改上面的例子,我們可以用None這個不變物件來實現:
def add_end(L=None):
if L is None:
L = []
L.append('END')
return L
現在,無論呼叫多少次,都不會有問題:
>>> add_end()
['END']
>>> add_end()
['END']
為什麼要設計str、None這樣的不變物件呢?因為不變物件一旦建立,物件內部的資料就不能修改,這樣就減少了由於修改資料導致的錯誤。此外,由於物件不變,多工環境下同時讀取物件不需要加鎖,同時讀一點問題都沒有。我們在編寫程式時,如果可以設計一個不變物件,那就儘量設計成不變物件。
作者:五秋木
連結:https://www.jianshu.com/p/b690422ec02a
來源:簡書
簡書著作權歸作者所有,任何形式的轉載都請聯絡作者獲得授權並註明出處。