鍍金池/ 問答/Python/ Python的淺拷貝與is的問題

Python的淺拷貝與is的問題

>>> names = [1,2,3]
>>> names = [1,2,[3,4]]
>>> n = names.copy()
>>> n is names
False

is是比較 變量中存儲的內(nèi)存地址嗎?這樣的話,n和names中存儲的內(nèi)存地址不應(yīng)該是相同的嗎?
這個知識點(diǎn)一直沒能理解...
求助...

回答
編輯回答
久礙你

(1) is是比較兩個引用是否指向了同一個對象(引用比較)。

clipboard.png

所以,在修改a的同時,b隨之改變,這也是毋庸置疑的。如圖:

clipboard.png

(2) n和copy中的地址不相同。你之所以認(rèn)為相同是因?yàn)槟惆褱\拷貝和直接賦值(也有人叫它淺拷貝的一種形式)形式的運(yùn)算搞混了,或者說你還沒有理清楚它們的區(qū)別。

例如剛才的“b = a” 就是直接賦值(引用操作),而你的n = names.copy()是淺拷貝。

淺拷貝:拷貝了最外圍的對象本身,內(nèi)部的元素都只是拷貝了一個引用而已。也就是,把對象復(fù)制一遍,但是該對象中引用的其他對象我不復(fù)制。(或者說只拷貝父對象,不會拷貝對象的內(nèi)部的子對象。)

這句話一出,你可能會馬上明白,但你也可能一臉懵逼,舉個例子來解釋一下:

clipboard.png

做個圖解可能更方便理解:

clipboard.png

步驟詳解:
(1)創(chuàng)建兩個列表a,b。
(2)創(chuàng)建列表c,它的c[0],c[1]分別保存的是a,b的引用值。
(3)通過"e = c.copy()"拷貝一份出來,記?。含F(xiàn)在只拷貝了父層,父層只有c[0],c[1],那現(xiàn)在就只拷貝這一層對象,而c[0],c[1]指向的內(nèi)容,我不復(fù)制(請好好理解這句話)
(4)e的第一層(e[0],e[1])分別指向了列表a,b.從圖中看,e和c的內(nèi)容相等(也確實(shí)相等)。但是e,c的箭頭并沒有指向同一個對象?。。在c.copy()的時候開辟了空間,所以導(dǎo)致你的is比較返False.如圖:

clipboard.png

大概就解釋到這,小生不才,言語表達(dá)能力可能不是很難好,希望能幫到你。

2018年8月4日 12:15
編輯回答
臭榴蓮

is比較兩個變量是否指向同一對象,copy()之后生成了新的對象,所以is就返回False了

2018年7月25日 17:36
編輯回答
不二心

>>> a = 1
>>> names = [a]
>>> n = names.copy()
>>> n is names
False
>>> n[0] is names[0]
True
n和names中存儲的內(nèi)存地址不應(yīng)該是相同的嗎?

首先要明確你想對n和names比較,還是兩者內(nèi)部元素的比較。

names由于是一個list(mutable),copy操作才為n開辟了內(nèi)存空間,所以這里n和names用is比較是False。

而另一種immutable的copy情況:

>>> import copy
>>> a = (1, 2)
>>> copy.copy(a) is a
True
2017年8月1日 09:17