成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Python解惑:整數比較

開發 后端
在 Python 中一切都是對象,毫無例外整數也是對象,對象之間比較是否相等可以用==,也可以用is。清楚is和==的區別之后,對此也許你有可能會遇到下面的這些困惑,于是就有了這樣一篇文章,試圖把Python中一些隱晦的東西趴出來,希望對你有一定的幫助。

[[196562]]

在 Python 中一切都是對象,毫無例外整數也是對象,對象之間比較是否相等可以用==,也可以用is。==和is操作的區別是:

  • is比較的是兩個對象的id值是否相等,也就是比較倆對象是否為同一個實例對象,是否指向同一個內存地址。
  • ==比較的是兩個對象的內容是否相等,默認會調用對象的__eq__()方法。

清楚is和==的區別之后,對此也許你有可能會遇到下面的這些困惑,于是就有了這樣一篇文章,試圖把Python中一些隱晦的東西趴出來,希望對你有一定的幫助。我們先來看兩段代碼:

片段一:

  1. >>> a = 256 
  2.  
  3. >>> b = 256 
  4.  
  5. >>> a == b 
  6.  
  7. True 
  8.  
  9. >>>  

片段二:

  1. >>> a = 256 
  2.  
  3. >>> b = 256 
  4.  
  5. >>> a is b 
  6.  
  7. True 
  8.  
  9. >>>  

在交互式命令行執行上面兩段代碼,代碼片段一中的a==b返回True很好理解,因為兩個對象的值都是256,對于片段二,a is b也返回True,這說明a和b是指向同一個對象的,可以檢查一下他們的id值是否相等:

  1. >>> id(a) 
  2.  
  3. 8213296 
  4.  
  5. >>> id(b) 
  6.  
  7. 8213296 
  8.  
  9. >>>  

結果證明他倆的確是同一個對象,指向的是同一個內存地址。那是不是所有的整數對象只要兩個對象的值(內容)相等,它們就是同一個實例對象呢?換句話說,對于整數對象只要==返回True,is操作也會返回True嗎?帶著這個問題來看下面這兩段代碼:

片段一:

  1. >>> a = 257 
  2.  
  3. >>> b = 257 
  4.  
  5. >>> a == b 
  6.  
  7. True 
  8.  
  9. >>>  

片段二:

  1. >>> a = 257 
  2.  
  3. >>> b = 257 
  4.  
  5. >>> a is b 
  6.  
  7. False 
  8.  
  9. >>>  

對于257,a is b返回的竟然是False,結果可能在你的意料之中,也有可能出乎你的意料,但不管怎么,我們還是要刨根問底,找出問題的真相。

解惑一

出于對性能的考慮,Python內部做了很多的優化工作,對于整數對象,Python把一些頻繁使用的整數對象緩存起來,保存到一個叫small_ints的鏈表中,在Python的整個生命周期內,任何需要引用這些整數對象的地方,都不再重新創建新的對象,而是直接引用緩存中的對象。Python把這些可能頻繁使用的整數對象規定在范圍[-5, 256]之間的小對象放在small_ints中,但凡是需要用些小整數時,就從這里面取,不再去臨時創建新的對象。因為257不再小整數范圍內,因此盡管a和b的值是一樣,但是他們在Python內部卻是以兩個獨立的對象存在的,各自為政,互不干涉。

弄明白***個問題后,我們繼續在Python交互式命令行中寫一個函數,再來看下面這段代碼:

片段一:

  1. >>> c = 257 
  2.  
  3. >>> def foo(): 
  4.  
  5. ... a = 257 
  6.  
  7. ... b = 257 
  8.  
  9. ... print a is b 
  10.  
  11. ... print a is c 
  12.  
  13. ... 
  14.  
  15. >>> foo() 
  16.  
  17. True 
  18.  
  19. False  

呃,什么情況,是的,你沒看錯,片段一中的這段代碼 a、b 值都是257的情況下,出現了a is b返回True,而a is c 返回的False,a、b、c的值都為257,為什么會出現不同的結果呢?這對于剛剛好不容易建立起來的認知就被徹底否決了嗎,那這段代碼中究竟發生了什么?難道解惑一中的結論是錯誤的嗎?

解惑二

A Python program is constructed from code blocks. A block is a piece of Python program text that is executed as a unit. The following are blocks: a module, a function body, and a class definition. Each command typed interactively is a block. A script file (a file given as standard input to the interpreter or specified as a command line argument to the interpreter) is a code block. A script command (a command specified on the interpreter command line with the ‘-c‘ option) is a code block. structure-of-a-program

為了弄清楚這個問題,我們有必要先理解程序代碼塊的概念。Python程序由代碼塊構成,代碼塊作為程序的一個最小基本單位來執行。一個模塊文件、一個函數體、一個類、交互式命令中的單行代碼都叫做一個代碼塊。在上面這段代碼中,由兩個代碼塊構成,c = 257作為一個代碼塊,函數foo作為另外一個代碼塊。Python內部為了將性能進一步的提高,凡是在一個代碼塊中創建的整數對象,如果存在一個值與其相同的對象于該代碼塊中了,那么就直接引用,否則創建一個新的對象出來。Python出于對性能的考慮,但凡是不可變對象,在同一個代碼塊中的對象,只有是值相同的對象,就不會重復創建,而是直接引用已經存在的對象。因此,不僅是整數對象,還有字符串對象也遵循同樣的原則。所以 a is b就理所當然的返回True了,而c和a不在同一個代碼塊中,因此在Python內部創建了兩個值都是257的對象。為了驗證剛剛的結論,我們可以借用dis模塊從字節碼的角度來看看這段代碼。

  1. >>> import dis 
  2.  
  3. >>> dis.dis(foo) 
  4.  
  5.   2           0 LOAD_CONST               1 (257) 
  6.  
  7.               3 STORE_FAST               0 (a) 
  8.  
  9.   
  10.  
  11.   3           6 LOAD_CONST               1 (257) 
  12.  
  13.               9 STORE_FAST               1 (b) 
  14.  
  15.   
  16.  
  17.   4          12 LOAD_FAST                0 (a) 
  18.  
  19.              15 LOAD_FAST                1 (b) 
  20.  
  21.              18 COMPARE_OP               8 (is
  22.  
  23.              21 PRINT_ITEM           
  24.  
  25.              22 PRINT_NEWLINE       
  26.  
  27.   
  28.  
  29.   5          23 LOAD_FAST                0 (a) 
  30.  
  31.              26 LOAD_GLOBAL              0 (c) 
  32.  
  33.              29 COMPARE_OP               8 (is
  34.  
  35.              32 PRINT_ITEM           
  36.  
  37.              33 PRINT_NEWLINE       
  38.  
  39.              34 LOAD_CONST               0 (None) 
  40.  
  41.              37 RETURN_VALUE  

可以看出兩個257都是從常量池的同一個位置co_consts[1]獲取的。

總結

一番長篇大論之后,得出兩點結論:1、小整數對象[-5,256]是全局解釋器范圍內被重復使用,永遠不會被GC回收。2、同一個代碼塊中的不可變對象,只要值是相等的就不會重復創建新的對象。似乎這些知識點對日常的工作一點忙也幫不上,因為你根本不會用is來比較兩個整數對象的值是否相等。那為什么還要拿出來討論呢?嗯,程序員學知識,不應該淺嘗輒止,要充分發揮死磕到底的精神。 

責任編輯:龐桂玉 來源: Python開發者
相關推薦

2019-04-02 10:50:05

框架Spring開發

2010-07-14 16:48:02

Perl字符串比較

2016-10-14 15:00:45

2017-09-12 15:56:43

邊緣計算云計算架構

2009-07-20 10:36:29

什么是JDBC

2012-03-23 14:02:11

云計算

2011-04-07 09:40:57

DataReader鏈接關閉

2019-03-25 20:46:22

混合IT云計算私有云

2010-04-07 11:09:53

2018-06-06 10:14:32

Kafka時間輪任務

2020-08-13 19:25:43

Serverless安裝字體

2009-01-07 09:21:00

IIS服務器故障

2010-04-08 09:09:21

IT外包

2011-12-06 10:10:53

路由路由切換

2010-09-09 09:23:42

CSSDIV

2009-02-09 10:45:37

本地連接圖標不翼而飛行家解惑

2012-04-05 11:14:56

2009-08-26 14:27:54

C#委托和事件

2009-06-17 17:43:16

英特爾nehalem超線程

2014-09-25 10:53:59

rJavaR語言
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 精品一区二区三区在线观看国产 | 国产精品美女久久久久久久久久久 | 日韩欧美中文在线 | 欧美日韩综合一区 | 日韩中文字幕免费在线 | 视频一二三区 | 91毛片在线看 | 欧美在线视频免费 | 99pao成人国产永久免费视频 | 国产真实精品久久二三区 | 久久精品亚洲精品 | 亚洲国产一区在线 | 在线观看日本高清二区 | 中文一区二区 | 在线观看国产精品一区二区 | 婷婷免费视频 | 亚洲成人一级片 | 精品欧美一区二区三区精品久久 | 伊人网一区 | 日本一区二区在线视频 | 午夜天堂精品久久久久 | 久久精品免费观看 | 亚洲国产精品久久久 | 91免费在线 | 国产成人一区二区三区精 | 国产精品不卡视频 | 久久久久久一区 | 九九亚洲 | 九色91视频 | 国产一区二区在线免费观看 | zzzwww在线看片免费 | 久久精品二区 | 黄色大片免费网站 | 五月网婷婷 | 精品国产乱码久久久久久图片 | av影片在线 | 插插插干干干 | 欧美激情欧美激情在线五月 | 依人成人| 偷偷操视频| 欧美激情精品久久久久久变态 |