在Python中會用到對象之間比較,可以用==,也可以用is。但是它們的區(qū)別是什么呢?
is比較的是兩個(gè)實(shí)例對象是不是完全相同,它們是不是同一個(gè)對象,占用的內(nèi)存地址是否相同。萊布尼茨說過:“世界上沒有兩片完全相同的葉子”,這個(gè)is正是這樣的比較,比較是不是同一片葉子(即比較的id是否相同,這id類似于人的身份證標(biāo)識)。
==比較的是兩個(gè)對象的內(nèi)容是否相等,即內(nèi)存地址可以不一樣,內(nèi)容一樣就可以了。這里比較的并非是同一片葉子,可能葉子的種類或者脈絡(luò)相同就可以了。默認(rèn)會調(diào)用對象的__eq__()方法。
可以通過如下例子來區(qū)分比較下:
>>>a=["I","love","Python"]
>>>b=a
#a的引用復(fù)制給b,在內(nèi)存中其實(shí)是指向了用一個(gè)對象
>>>bisa
True
>>>id(a)
46381384
>>>id(b)
46381384
#當(dāng)然,內(nèi)容也肯定是相等的
>>>b==a
True
可以發(fā)現(xiàn)b和a的內(nèi)存地址是相同的,它們指向同一塊內(nèi)存,因而is和==的結(jié)果都為True。這是因?yàn)橹苯淤x值都是賦值的引用,是引用,是引用,重要的事情說三遍。但如果不是通過引用賦值,而是通過切片賦值呢?
#b通過切片操作重新分配了對象,但是值和a相同
>>>b=a[:]
>>>bisa
False
>>>id(a)
48740680
>>>id(b)
48740680
>>>b==a#但他們的值還是相等的
True
新建對象之后,b和a指向了不同的內(nèi)存,所以bisa的結(jié)果為False,而b==a的結(jié)果為True。在這里,小編提一個(gè)問題,b[0]isa[0]的結(jié)果呢?
答案是True。因?yàn)榍衅截愂菧\拷貝,列表中的元素并未重新創(chuàng)建。不理解的同學(xué)請翻看之前的文章Python中的淺拷貝與深拷貝。
通常,我們關(guān)注的是值,而不是內(nèi)存地址,因此Python代碼中==出現(xiàn)的頻率比is高。但是什么時(shí)候用is呢?
is與==相比有一個(gè)比較大的優(yōu)勢,就是計(jì)算速度快,因?yàn)樗荒苤剌d,不用進(jìn)行特殊的函數(shù)調(diào)用,少了函數(shù)調(diào)用的開銷而直接比較兩個(gè)整數(shù)id。而a==b則是等同于a.__eq__(b)。繼承自object的__eq__方法比較兩個(gè)對象的id,結(jié)果與is一樣。但是多數(shù)Python的對象會覆蓋object的__eq__方法,而定義內(nèi)容的相關(guān)比較,所以比較的是對象屬性的值。
在變量和單例值之間比較時(shí),應(yīng)該使用is。目前,最常使用is的地方是判斷對象是不是None。下面是推薦的寫法:
aisNone
判斷不是None的推薦寫法是:
aisnotNone
Python會對比較小的整數(shù)對象進(jìn)行緩存,下次用的時(shí)候直接從緩存中獲取,所以is和==的結(jié)果可能相同:
>>>a=1
>>>b=1
>>>aisb
True
>>>a==b
True
而看一下另外一段代碼:
>>>a=257
>>>b=257
>>>aisb
False
這是什么原因呢?
注意,Python僅僅對比較小的整數(shù)對象進(jìn)行緩存(范圍為范圍[-5,256])緩存起來,而并非是所有整數(shù)對象。需要注意的是,這僅僅是在命令行中執(zhí)行,而在Pycharm或者保存為文件執(zhí)行,結(jié)果是不一樣的,這是因?yàn)榻忉屍髯隽艘徊糠謨?yōu)化。
總結(jié)
1、is比較兩個(gè)對象的id值是否相等,是否指向同一個(gè)內(nèi)存地址;
2、==比較的是兩個(gè)對象的內(nèi)容是否相等,值是否相等;
3、小整數(shù)對象[-5,256]在全局解釋器范圍內(nèi)被放入緩存供重復(fù)使用;
4、is運(yùn)算符比==效率高,在變量和None進(jìn)行比較時(shí),應(yīng)該使用is。
以上內(nèi)容為大家介紹了Python中的比較:is與==,希望對大家有所幫助,如果想要了解更多Python相關(guān)知識,請關(guān)注IT培訓(xùn)機(jī)構(gòu):千鋒教育。