Python面試之 is 和 == 的區別

面試實習生的時候,當問到 is 和 == 的區別時,很多同學都答不上來,搞不清兩者什麼時候返回一致,什麼時候返回不一致。本文我們來看一下這兩者的區別。

我們先來看幾個例子:

a = "hello"b = "hello"print(a is b) # 輸出 True print(a == b) # 輸出 Truea = "hello world"b = "hello world"print(a is b) # 輸出 Falseprint(a == b) # 輸出 True a = [1, 2, 3]b = [1, 2, 3]print(a is b) # 輸出 Falseprint(a == b) # 輸出 True a = [1, 2, 3]b = aprint(a is b) # 輸出 True print(a == b) # 輸出 True

上面的輸出結果中為什麼有的 is 和 == 的結果相同,有的不相同呢?我們來看下官方文檔中對於 is 和 == 的解釋。

官方文檔中說 is 表示的是對象標示符(object identity),而 == 表示的是相等(equality)。is 的作用是用來檢查對象的標示符是否一致,也就是比較兩個對象在內存中的地址是否一樣,而 == 是用來檢查兩個對象是否相等。

我們在檢查 a is b 的時候,其實相當於檢查 id(a) == id(b)。而檢查 a == b 的時候,實際是調用了對象 a 的 __eq()__ 方法,a == b 相當於 a.__eq__(b)。

一般情況下,如果 a is b 返回True的話,即 a 和 b 指向同一塊內存地址的話,a == b 也返回True,即 a 和 b 的值也相等。

好了,看明白上面的解釋後,我們來看下前面的幾個例子

a = "hello"b = "hello"print(id(a)) # 輸出 140506224367496print(id(b)) # 輸出 140506224367496print(a is b) # 輸出 True print(a == b) # 輸出 Truea = "hello world"b = "hello world"print(id(a)) # 輸出 140506208811952print(id(b)) # 輸出 140506208812208print(a is b) # 輸出 Falseprint(a == b) # 輸出 True a = [1, 2, 3]b = [1, 2, 3]print(id(a)) # 輸出 140506224299464print(id(b)) # 輸出 140506224309576print(a is b) # 輸出 Falseprint(a == b) # 輸出 True a = [1, 2, 3]b = aprint(id(a)) # 輸出 140506224305672print(id(b)) # 輸出 140506224305672print(a is b) # 輸出 True print(a == b) # 輸出 True

列印出 id(a) 和 id(b) 後就很清楚了。只要 a 和 b 的值相等,a == b 就會返回True,而只有 id(a) 和 id(b) 相等時,a is b 才返回 True。

這裡還有一個問題,為什麼 a 和 b 都是 "hello" 的時候,a is b 返回True,而 a 和 b都是 "hello world" 的時候,a is b 返回False呢?

這是因為前一種情況下Python的字元串駐留機制起了作用。對於較小的字元串,為了提高系統性能Python會保留其值的一個副本,當創建新的字元串的時候直接指向該副本即可。所以 "hello" 在內存中只有一個副本,a 和 b 的 id 值相同,而 "hello world" 是長字元串,不駐留內存,Python中各自創建了對象來表示 a 和 b,所以他們的值相同但 id 值不同。(這段解釋有誤,非常感謝 @冒泡 同學指正。 @冒泡 同學指出:intern機制和字元串長短無關,在交互模式下,每行字元串字面量都會申請一個新字元串,但是只含大小寫字母、數字和下劃線的會被intern,也就是維護了一張dict來使得這些字元串全局唯一)

總結一下,is 是檢查兩個對象是否指向同一塊內存空間,而 == 是檢查他們的值是否相等。可以看出,is 是比 == 更嚴格的檢查,is 返回True表明這兩個對象指向同一塊內存,值也一定相同。

看到這裡,大家是不是搞懂了 is 和 == 的區別呢?

那我們深入一步來思考一下下面這個問題:

Python里和None比較時,為什麼是 is None 而不是 == None 呢?

歡迎回答在評論區~

推薦閱讀:

代碼優化指南:人生苦短,我用Python
Scrapy對接Splash
利用函數計算實現網路遊戲或視頻直播中的敏感詞檢測
(1 條消息)Python如何向控制台列印#號?
數據分析入門必看案例:泰坦尼克號倖存率研究

TAG:Python入門 | Python |