警惕python中的*重複符(運算子)

pythontab發表於2014-05-05

在python中有個特殊的符號“*”,可以用做數值運算的乘法運算元,也是用作物件的重複運算元,但在作為重複運算元使用時一定要注意

注意的是:*重複出來的各物件具有同一個id,也就是指向在記憶體中同一塊地址,在對各個物件進行操作是一定要注意。

舉例來說:

>>> alist = [range(3)]*4
>>> alist
[[0, 1, 2], [0, 1, 2], [0, 1, 2], [0, 1, 2]]

上面初始化一個二層列表用來模擬矩陣,該矩陣式4X3的,為描述方便,這裡記矩陣為A。

現在我想給A11賦值為1,用下面的程式碼:

alist[0][0]=1

那我們想要的結果應該是:

[[1, 1, 2], [0, 1, 2], [0, 1, 2], [0, 1, 2]]

可是很不幸,我們得到的是:

[[1, 1, 2], [1, 1, 2], [1, 1, 2], [1, 1, 2]]

這是怎麼回事呢,為什麼給A21賦值,其他的Ai1為什麼都跟著變了呢?


原因是這樣的:

文章開始的時候我們已經說過了,*重複出來的各物件具有同一個id,也就是指向在記憶體中同一塊地址,在對各個物件進行操作是一定要注意

我們再初始化的時候用了重複運算元"*",這個運算子在對物件進行重複操作時,會將重複的所有物件都指向同一塊記憶體地址,所有當你改變其中的一個值時,

其他的值自然也會更新,用python的話進行解釋就是下面的命令及輸出:

>>> id(alist[0])
18858192
>>> id(alist[1])
18858192
>>> id(alist[2])
18858192
>>> id(alist[3])
18858192
>>>

看到了吧,id都是一樣滴,也就是說這4個list是同一個“list”。

既然這樣那我們想要模擬一個矩陣怎麼辦呢,除了有專門的numpy包之外,你當然可以給上層list逐個的append新的list,例如:

>>> blist=[]
>>> for i in range(4):
    blist.append([j for j in range(3)])
>>> blist
[[0, 1, 2], [0, 1, 2], [0, 1, 2], [0, 1, 2]]

這樣,我們再試試上面的賦值操作:

>>> blist[0][0]=1
>>> blist
[[1, 1, 2], [0, 1, 2], [0, 1, 2], [0, 1, 2]]
>>>


相關文章