Python初學(xué)者:“==”and“is”的區(qū)別是什么?
本文轉(zhuǎn)載自公眾號(hào)“讀芯術(shù)”(ID:AI_Discovery)。
幾天前,我在Reddit上瀏覽“learn python”板塊的相關(guān)內(nèi)容時(shí),看到一位Reddit用戶再次提問了這個(gè)問題:“==” and “is”的區(qū)別是什么?雖然網(wǎng)上對(duì)此問題已經(jīng)有太多的答案和解釋,但是很多初學(xué)者還是不知道,還是會(huì)犯錯(cuò)。
“==”和“is”都是Python中的運(yùn)算符。初學(xué)者可能會(huì)把“a == b”理解為“a等于b”,而把“a is b” 理解為 “a is b”。也許這就是Python初學(xué)者混淆“==”和“is”的原因。
在深入討論之前,我想先舉幾個(gè)“==” 和 “is”的用例:
- >>> a = 5
- >>> b = 5
- >>> a == bTrue>>> a is b
- True
簡(jiǎn)單吧?a== b 和 a is b 都能返回 True。下一個(gè)例子:
- >>> a = 1000
- >>> b = 1000
- >>> a == bTrue>>> a is b
- False
這是為什么?第二個(gè)例子與第一個(gè)唯一的不同就在于a和b的值從5變成了1000,但是“==” 和 “is”輸出的結(jié)果卻完全不同。再看下一例:
- >>> a = []
- >>> b = []>>> a == bTrue>>> a is b
- False
如果這還不夠震驚,再看最后一個(gè)例子:
- >>> a = 1000
- >>> b = 1000
- >>> a == bTrue>>> a is b
- False>>> a = b>>> a == bTrue>>> a is b
- True
“==”的正式運(yùn)算是相等,而“is”的運(yùn)算是標(biāo)識(shí)。用“==”是比較兩個(gè)對(duì)象的值。“a == b”應(yīng)解釋為“a的值是否等于b的值”。在上述所有示例中,a的值始終等于b的值(即使對(duì)于空列的示例也是如此),因此“a == b”始終為真。
在解釋標(biāo)識(shí)的概念之前,我需要先介紹一下id函數(shù)。對(duì)象的標(biāo)識(shí)可以通過id函數(shù)來獲得。一個(gè)對(duì)象的標(biāo)識(shí)始終是唯一且恒定的,你可以將其視為該對(duì)象的地址。如果兩個(gè)對(duì)象的標(biāo)識(shí)相同,則它們的值也一定相同。
- >>> id(a)
- 2047616
運(yùn)算符“is”用于比較兩個(gè)對(duì)象的標(biāo)識(shí)是否相同,“a is b”就表示“a的標(biāo)識(shí)與b的標(biāo)識(shí)相同”。
圖源:unsplash
現(xiàn)在你知道了“==”和“is”的真正含義,我們就可以開始深入討論上述示例。
首先是第一例和第二例的結(jié)果差異。因?yàn)镻ython存儲(chǔ)了一個(gè)介于-5到256之間的整數(shù)數(shù)組列表,每一個(gè)整數(shù)都有固定的對(duì)應(yīng)標(biāo)識(shí)。當(dāng)你在此范圍內(nèi)分配整數(shù)變量時(shí),Python就會(huì)將此變量作為數(shù)組列里的整數(shù)為其分配標(biāo)識(shí)。
因此,在第一例中,由于a和b的標(biāo)識(shí)都是從數(shù)組列表中獲得的,所以他們的標(biāo)識(shí)當(dāng)然相同,因此a is b為真。
- >>> a = 5
- >>> id(a)
- 1450375152
- >>> b = 5
- >>> id(b)
- 1450375152
但一旦變量的值不在這個(gè)區(qū)間范圍內(nèi),由于Python內(nèi)部沒有對(duì)應(yīng)該值的對(duì)象,因此Python將為此變量創(chuàng)建新的標(biāo)識(shí),并為這個(gè)變量賦值。
如前所述,每個(gè)創(chuàng)建的標(biāo)識(shí)都是唯一的,因此即使兩個(gè)變量的值相同,他們的標(biāo)識(shí)也永遠(yuǎn)不會(huì)等同。這就是為什么第二例中的a is b返回False。
- >>> a = 1000
- >>> id(a)
- 12728608
- >>> b = 1000
- >>> id(b)
- 13620208
另外,假設(shè)你打開的是兩個(gè)控制臺(tái),如果該值仍在區(qū)間內(nèi),也能得到相同標(biāo)識(shí)。但是,如果該值不在區(qū)間內(nèi),結(jié)果當(dāng)然就不同了。
一旦理解了第一例和第二例的區(qū)別,就很容易理解第三例的結(jié)果了。由于Python不存儲(chǔ)“空列表”對(duì)象,所以Python創(chuàng)建了一個(gè)新對(duì)象并賦值“空列表”。無論這兩個(gè)列表是空還是元素相同,結(jié)果都是一樣的。
- >>> a = [1,10,100,1000]
- >>> b = [1,10,100,1000]
- >>> a == bTrue>>> a is bFalse>>> id(a)
- 12578024
- >>> id(b)
- 12578056
來看最后一例。第二例與最后一例的唯一區(qū)別在于多了一行代碼a = b。然而,這行代碼卻改變了變量a的命運(yùn)。下面的結(jié)果將闡述原因:
- >>> a = 1000
- >>> b = 2000
- >>> id(a)
- 2047616
- >>> id(b)
- 5034992
- >>> a = b>>> id(a)
- 5034992
- >>> id(b)
- 5034992
- >>> a2000>>> b2000
可以看到,在a= b之后,a的標(biāo)識(shí)變成了b的標(biāo)識(shí)。a = b把b的標(biāo)識(shí)賦予了a。因此a和b就擁有了相同的標(biāo)識(shí),a的值現(xiàn)在就等于b的值,即2000。
最后一例傳達(dá)出一個(gè)重要信息,即你可能在不經(jīng)意間更改了對(duì)象的值,尤其是當(dāng)對(duì)象為列表時(shí)。
- >>> a = [1,2,3]
- >>> id(a)
- 5237992
- >>> b = a
- >>> id(b)
- 5237992
- >>> a.append(4)
- >>> a
- [1, 2, 3, 4]
- >>> b
- [1, 2, 3, 4]
從上例可以看出,a和b擁有相同的標(biāo)識(shí),他們的值就一定相同。因此在為a附加了一個(gè)新元素后,b的值也會(huì)受到影響。為了避免這種情況,如果要把一個(gè)對(duì)象的值復(fù)制到另一對(duì)象,又不引用同一標(biāo)識(shí),一個(gè)方法是在copy模塊中使用deepcopy。對(duì)于列表,你還可以通過b= a[:]來實(shí)現(xiàn)。
- >>> import copy
- >>> a = [1,2,3]
- >>> b= copy.deepcopy(a)
- >>> id(a)
- 39785256
- >>> id(b)
- 5237992
使用[:]把元素復(fù)制到新變量:
- >>> a = [1,2,3]
- >>> id(a)
- 39785256
- >>> b = a[:]
- >>> id(b)
- 23850216
- >>> a.append(4)
- >>> a
- [1, 2, 3, 4]
- >>> b
- [1, 2, 3]
希望這篇文章能幫你徹底解決這個(gè)問題,不要再被相同的難題困住啦。