电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

詳解Python垃圾回收機制和常量池的驗證

瀏覽:22日期:2022-06-26 08:03:27
Python的引入

人類認識世界是從認識世界中的一個又一個實物開始,然后再對其用語言加以描述。例如當中國人看到蘋果時,便會用中文“蘋果”加以描述,而用英語的一些國家則會用“apple”加以描述。

以上說到的中文和英文都是人類認識并描述世界的一個工具,而在計算機的世界中,為了讓計算機去認知世界,從而幫助人類完成更多的任務。在計算機領域中也發展了語言這個工具,從早期的機器語言到匯編語言再到現在使用范圍較廣的高級語言。而我們接下來要介紹的Python則屬于高級語言這一分支。

變量的引入為什么要有變量

上面說到Python是計算機世界中用來描述外部世界的,并且也提及了世界就是一個又一個實物的堆疊,描述世界其實就是去描述那一個又一個實物,人類如此,計算機也是如此。因此計算機語言開發者們為了使用計算機語言的人更好的在計算機中去描述這些實物,便在計算機語言中引入了變量這個概念,Python也不例外。簡單點說,變量就是用來描述世間萬物的。

定義變量

為了在計算機書寫方便,定義一變量也有一定的規則,在這里我們僅說說Python中變量的定義規則,首先我們先定義兩個變量:

name = ’chenyoude’year = 2021

上述代碼中我們便定義了兩個變量,從上面定義的兩個變量中,我們可以看到,變量的組成分為三個部分:

變量名:反應變量值所描述的意義,并且可以用來引用變量值。 賦值符號:賦值。 變量值:存放數據,用來記錄現實世界中的某種狀態。 常量引入

上面簡單講解了Python中的變量,通過字面意思,可以看到變量其實是一個變化的量,例如,下面這個實例:

year = 2021year = year + 1print(year) # 輸出結果:2022

剛開始我們賦予了year一個變量值為2021,當我們對year進行加1操作時,可以發現year值變成了2022。對于上述現象我們不難理解,因為之前說過Python中變量是用來描述世間萬物的,世間萬物在現實中是可以變化的,變量當然也可以隨之變化。

但是在某個局部范圍內,變量可能是不會變化的,例如在2021年這一年,都只會是2021年,沒有人會說2021年是2022年。如果你有豐富的開發經驗,會明白變量定義出來不是存放在那里給你看的,更多的是要拿來用的。也就是說如果在2021年中的某個程序需要使用year這個變量,但這個變量是不需要進行修改的。為了防止誤操作對year這個變量進行了修改,計算機語言便設計了常量這個概念,也就是說常量相對于變量是一個不會變化的量。

在Python中,有沒有常量呢?不嚴格的講,其實是有的,只是在定義常量的時候常量名必須的全大寫,例如,下面這個實例:

YEAR = 2021YEAR = YEAR + 1print(YEAR) # 輸出結果:2022

上面這個常量的實例令人大吃一驚,因為使用常量YEAR后和使用變量year的結果一致,也就是說常量YEAR遭到了更改。但是,稍微解釋你就明白了。

在Python中,雖然也和其他很多計算機語言一樣擁有常量這個概念,但更多的是約定俗成的,Python并沒有嚴格的對常量進行控制,只是規定常量名必須全部大寫。原因很簡單:都是常量了,你為什么還要修改?

常量池引入

上面講到常量就是一個不會變化的變量,嚴格的講,在Python中是沒有常量這個概念的。但是,在Python中又有另外一種例外,那就是常量池,為了搞清楚常量池,首先我們得弄明白Python的幾個小知識,接下來一一敘說。

Python解釋器

上面提及到Python是計算機用來描述世間萬物的一種語言,由于計算機沒有人腦那么強大,計算機更多的只是認識高低壓電頻,再通過對高低壓電頻的轉化進而編碼成我們看到的一個又一個字符,也就是說計算機是無法直接認識利用Python寫下的字符的。(此處設計計算機組成原理,不多做介紹)

也就是說,當我們利用Python寫下一個又一個字符并且交給電腦時,需要通過編碼這個過程,而這個編碼的過程有時候也被稱為解釋。解釋的原理就相當于從中文轉成英文,只不過此時不是需要讓英文使用者看懂中文,而是讓計算機能夠看懂Python。

中文轉成英文的時候,可能需要一個翻譯員或一個翻譯軟件,利用Python寫下的字符轉化為計算機能看懂的語言同樣如此,這個轉化過程也需要一個外物的幫助——Python解釋器。

Python變量存儲機制

假設我們使用Python解釋器定義了以下一個變量:

year = 2021

當我們通過字符定義變量時,一定會好奇這些變量被Python解釋器解釋后到底去了哪?如果對計算機的組成熟悉的同學,一定會清楚計算機的核心組件為:CPU、內存、外存、輸入設備、輸出設備。也就是說,這些字符應該存儲在這些核心組件中。在這里就不賣關子了,當我們通過字符定義變量并對其用Python解釋器進行解釋時,他們會以計算機能看懂的形式進入內存當中。

上面講的對于很多非科班出身的朋友可能很難理解,在這里將它生動化。現在假設江西師范大學相當于電腦內存,每當有一批新學生進入師大時,師大都會開辟出一個新教室給這批新同學使用,并且會給每一個教室一個獨一無二的教室牌號。由于把師大看作是內存,這批新同學就可以看成是變量值,而教室牌號就是變量名。也就是說,對于師大這個大內存,每定義一個變量year=2021,就會在這個大內存中開辟一個小空間,小空間中放變量值2021,然后大內存會給這個小空間定義一個變量名year,此時變量名year指向變量值2021。

上面說到每當Python解釋器解釋一個變量時,會將這個變量存放到內存中的一個小空間中,但如何知道這個小空間的具體位置呢?此處介紹Python的一個內置函數id(),通過這個函數可以獲取某一個變量所在的內存地址,例如下面這個實例:

year = 2021print(id(year)) # 輸出4499932432Python垃圾回收機制

對于上述師大的例子,此處再做延伸。由于那一批學生所在班級新轉來了幾位同學,需要那一批學生更換更大一點教室,也就是給他們一個新的教室。那么學校應該會這樣處理,首先開辟一個新的教室,然后拿下那一批學生原有教室的教室牌號更換到這個新教室,最后會清空原有教室。

在Python中,也是如此,如果到了新的一年,我們會重新定義一個year變量,也就是year=2022。如果這是在同一個程序中如此做,Python會沿用上述更換教室的方法,它首先會解除year和2021的連接,開辟一個新內存存放變量值2022,讓year與2022連接。此時,會發現2021這個變量值只有變量值而沒有變量名,因此這個沒有變量名的變量值會變成Python眼中的一個垃圾變量,從而觸發Python垃圾回收機制,對這個2021所在的內存空間進行回收。

為了更好地理解Python垃圾回收機制,可以看下面這個例子:

year = 2021print(id(year)) # 輸出4499932720print(year) # 輸出2021year = 2022print(id(year)) # 輸出4499932560print(year) # 輸出2022

通過上述例子,可以看到當新定義了一個year變量時,year會與新的變量進行一個連接。當然,此處所說的垃圾回收機制只是為了引入引用計數這個概念,并不是完全正確的解釋,并且上述實例還無法證明變量值2021所在內存是否被回收,下面將通過引用計數的實例會進一步說明并重新解釋垃圾回收機制。

引用計數

上面講到如果某個變量值綁定著變量名,就是一個正常的變量,如果該變量值沒有綁定著門牌號,這個變量就是一個垃圾變量,對于垃圾變量,Python會觸發垃圾回收機制回收這個變量所占有的內存。進而可以想到,Python中一個變量名一定只能對應一個變量值。

在這里我們就不能沿用師大這個例子了,而得引出一個新的名詞——引用計數。

為了解釋引用計數,我們首先得明白在Python中,當定義了一個變量值為2021的變量時,它可以表示年份、也可以表示山的高度…也就是說一個變量名只能對應一個變量值,但是一個變量值可以對應不同的變量名,這種設計也是比較合理的。

現在我們引出引用計數這個概念,當相同的變量值被賦予不同的變量名時,變量值每增加一個變量名的賦予,則該變量值的引用計數加1。由于我們可以通過Python內置sys模塊中的getrefcount()函數獲取某一個變量的引用計數(getrefcount輸出值默認從3開始),可以通過下面這個例子感受下:

import sys# 引用計數初始值為3print(sys.getrefcount(2021)) # 輸出為3year = 2021print(sys.getrefcount(2021)) # 輸出為4height = 2021print(sys.getrefcount(2021)) # 輸出為5del yearprint(sys.getrefcount(2021)) # 輸出為4

從上述代碼可以看出變量值2021的引用計數由于每一次賦予新的變量名,引用計數都會增加,而當我們利用del關鍵字刪除變量值2021的一個變量名year時,引用計數則會減少。

為了更加嚴謹的表達引用計數,此處不得不再次深入,引用計數字面意思可以理解為引用的次數,也就是說上面的例子其實并不嚴謹,更嚴謹的講,只有當一個變量值每一次被直接或間接引用時,引用計數才會增加,在Python中讓引用計數增加共有三種方法:

變量被創建,變量值引用計數加1 變量被引用,變量值引用計數加1 變量作為參數傳入到一個函數,變量值引用計數加2

具體看下述實例:

import sys# 引用計數初始值為3print(sys.getrefcount(2021)) # 輸出為3# 變量被創建,變量值引用計數加1year = 2021print(sys.getrefcount(2021)) # 輸出為4# 變量被引用,變量值引用計數加1height = yearprint(sys.getrefcount(2021)) # 輸出為5# 變量作為參數傳入到一個函數,變量值引用計數加2def func(year): print(sys.getrefcount(year)) func(year) # 輸出為7

Python中既然有增加引用計數的方法, 也當然會減少引用計數的方法,共有以下4種:

變量值對應的變量名被銷毀 變量值對應的變量名被賦予新的值 變量值對應的變量名離開它的作用域 變量值對應的變量名的容器被銷毀 重看Python垃圾回收機制

有了getrefcount()方法并通過引用計數,我們就可以解開垃圾回收機制遺留的一個問題——如何判斷是否觸發了垃圾回收機制。每當一個變量定義,他的getrefcount輸出值為3,而如果該變量值被垃圾回收機制回收,則它的getrefcount輸出值回到3,可以通過下面實例驗證上述猜想:

import sysprint(sys.getrefcount(2021)) # 輸出為3year = 2021print(sys.getrefcount(2021)) # 輸出為4print(id(year)) # 輸出4499932720print(year) # 輸出2021year = 2022print(sys.getrefcount(2021)) # 輸出為3print(id(year)) # 輸出4499932560print(year) # 輸出2022

通過上述實例,可以發現由于變量值2021對應的變量名被新的變量值2022引用,它的getrefcount輸出值為3,引用計數變成了0,因此可以證明Python觸發了垃圾回收機制。

如果對上述驗證Python觸發垃圾回收機制的實例深入挖掘,會發現當把year賦給變量值2022時,變量值的2021的引用計數為0,此時觸發了Python的垃圾回收機制,那么是否可以表明只有當變量值2021的引用計數為0時才能觸發垃圾回收機制呢?而不是上一次說的當變量值的變量名被新的變量值被引用了才會銷毀呢?因為變量值可以對應多個變量名,下面通過下述實例驗證:

import sysprint(sys.getrefcount(2021)) # 輸出為3year = 2021print(sys.getrefcount(2021)) # 輸出為4height = 2021print(sys.getrefcount(2021)) # 輸出為5year = 2022print(sys.getrefcount(2021)) # 輸出為4del heightprint(sys.getrefcount(2021)) # 輸出為3

通過上述實例,可以發現由于定義一個變量后,該變量對應的變量值引用計數可以不斷增加,而只要引用計數不為0,那么Python就一直還在內存中保留著這個變量值并且對其引用,只有當該變量的引用計數為0時,Python才會觸發垃圾回收機制對該變量值進行回收,這才是比較正確的垃圾回收機制。當然,如果深入,Python的回收機制還有分代回收,此處不做延展,了解上述這些就足矣了解接下來講的小整數池。

常量池

在上述各個知識的打通之后,現在可以正式引入常量池這個概念。上面講到在Python中嚴格的講是沒有常量這個概念的,即使你通過約定俗成的方法定義了一個常量,但這個常量也只是一個變量,也就是說只要你對這個常量做出修改,這個常量原有對應的常量值引用計數就會變成0,由于常量等同于變量,它一樣會被Python垃圾回收機制回收。

但是在Python中,存在著一些例外,這些例外就是一個小整數池,顧名思義,小整數池表示的是從-5到256范圍內的整數,這些整數定義出來后就是一個常量,也就是說他們的引用計數即使為0,也不會被Python的垃圾回收機制回收,可以通過下述實例驗證:

import sysfirst_l = [] # 定義列表l存儲[-5,256]中的所有整數的引用計數add_l = [] # 定義列表add_l存儲[-5,256]中的所有整數的引用計數加1后的引用計數del_l = [] # 定義列表del_l存儲[-5,256]中的所有整數的引用計數減1后的引用計數for i in range(-5, 256): first_l.append(sys.getrefcount(i)) add = i add_l.append(sys.getrefcount(i)) del add del_l.append(sys.getrefcount(i))first_l.sort()add_l.sort()del_l.sort()print(f’min(first_l): {min(first_l)}’) # 獲取[-5,256]中所有整數的最小引用計數,輸出為4print(f’min(add_l): {min(add_l)}’) # 獲取[-5,256]中所有整數的最小引用計數,輸出為5print(f’min(del_l): {min(del_l)}’) # 獲取[-5,256]中所有整數的最小引用計數,輸出為4

從上述實例可以看出,[-5,256]中的整數的getrefcount默認初始值為4,也就是說即使沒有對這些整數進行初始化的創建,Python早已對他們進行了引用,即使他們的引用計數為0,他們也不會也不可能被刪除,因為他們從Python解釋器啟動開始就已經被生成。

當然,也可以通過垃圾回收機制判斷小整數池中的整數是否會被垃圾回收機制回收,可用如下實例證明(由于Pycharm等解釋器會一次性編譯整個文件,固使用終端編輯代碼):

>>> a = 5>>> id(a)4529334480>>> del a>>> b = 5>>> id(b)4529334480>>>>>> a = 257>>> id(a)4533920752>>> del a>>> b = 257 # 消除分代回收對結果的影響>>> del b>>> b = 257>>> id(b)4531031792>>>

從上述實例中可以看出,變量值5即使被垃圾回收機制回收后,再次創建變量值為5的變量,該變量的內存地址始終無變化,即該變量未被垃圾回收機制回收,小整數池中的其他整數同理;而變量值257卻已經被垃圾回收機制回收,非小整數池中的其他變量同理。

當然,還可以通過下述方法查看這些小整數池的整數的內存地址的變化,如下:

a = 256b = int('256')print(id(a), id(b)) # 4544968752 4544968752a = 257b = int('257')print(id(a), id(b)) # 4548719792 4546289360a = -5b = int('-5')print(id(a), id(b)) # 4544960400 4544960400a = -6b = int('-6')print(id(a), id(b)) # 4690036912 4546289360

對于上述實例,在Python中,由于每生成一個變量便會開辟一個新的內存空間給該變量,但是上述實例表明當變量值為-5和256時,每次開辟的內存空間地址都是一樣的;而當變量值不屬于[-5,256]時,每次定義變量值時,內存空間的地址都是不一樣的。

總結

在Python中,變量是用來描述世間萬物的,變量顧名思義是變化的一個量,而在某一個局部范圍內,有些量可能是不會變化的,因此語言設計者在計算機中定義了常量這個概念,但是在Python中并沒有規定的常量,只有約定俗稱的常量,也就是變量名全大寫的則是常量。但是Python中有一個另外,也就是小整數池[-5,256],在這個小整數池中的整數對于Python來說就是一個常量,因為從引用計數的打印中可以看出它在Python解釋器啟動的時候就已經生成并占用了一個固定的內存空間,并且不會因為引用計數變為0之后就會被Python的垃圾回收機制回收,而這些小整數池也可以稱作Python的常量池。

以上就是詳解Python垃圾回收機制和常量池的驗證的詳細內容,更多關于Python垃圾回收機制和常量池的驗證的資料請關注好吧啦網其它相關文章!

標簽: Python 編程
相關文章:
主站蜘蛛池模板: 驾驶人在线_专业学车门户网站 | 铸铁平台,大理石平台专业生产厂家_河北-北重机械 | 双能x射线骨密度检测仪_dxa骨密度仪_双能x线骨密度仪_品牌厂家【品源医疗】 | 金属波纹补偿器厂家_不锈钢膨胀节价格_非金属伸缩节定制-庆达补偿器 | 南京交通事故律师-专打交通事故的南京律师| 菏泽商标注册_菏泽版权登记_商标申请代理_菏泽商标注册去哪里 | 钛合金标准件-钛合金螺丝-钛管件-钛合金棒-钛合金板-钛合金锻件-宝鸡远航钛业有限公司 | 企业微信scrm管理系统_客户关系管理平台_私域流量运营工具_CRM、ERP、OA软件-腾辉网络 | 魔方网-培训咨询服务平台| 英思科GTD-3000EX(美国英思科气体检测仪MX4MX6)百科-北京嘉华众信科技有限公司 | 滁州高低温冲击试验箱厂家_安徽高低温试验箱价格|安徽希尔伯特 | 耐酸泵,耐腐蚀真空泵,耐酸真空泵-淄博华舜耐腐蚀真空泵有限公司 精密模具-双色注塑模具加工-深圳铭洋宇通 | 采暖炉_取暖炉_生物质颗粒锅炉_颗粒壁炉_厂家加盟批发_烟台蓝澳采暖设备有限公司 | 安规_综合测试仪,电器安全性能综合测试仪,低压母线槽安规综合测试仪-青岛合众电子有限公司 | 雨水收集系统厂家-雨水收集利用-模块雨水收集池-徐州博智环保科技有限公司 | 苗木价格-苗木批发-沭阳苗木基地-沭阳花木-长之鸿园林苗木场 | 硬度计_影像测量仪_维氏硬度计_佛山市精测计量仪器设备有限公司厂家 | 济南网站建设|济南建网站|济南网站建设公司【济南腾飞网络】【荐】 | 临海涌泉蜜桔官网|涌泉蜜桔微商批发代理|涌泉蜜桔供应链|涌泉蜜桔一件代发 | 二手光谱仪维修-德国OBLF光谱仪|进口斯派克光谱仪-热电ARL光谱仪-意大利GNR光谱仪-永晖检测 | YT保温材料_YT无机保温砂浆_外墙保温材料_南阳银通节能建材高新技术开发有限公司 | 十字轴_十字轴万向节_十字轴总成-南京万传机械有限公司 | 铁艺,仿竹,竹节,护栏,围栏,篱笆,栅栏,栏杆,护栏网,网围栏,厂家 - 河北稳重金属丝网制品有限公司 山东太阳能路灯厂家-庭院灯生产厂家-济南晟启灯饰有限公司 | 翰墨AI智能写作助手官网_人工智能问答在线AI写作免费一键生成 | 地源热泵一体机,地源热泵厂家-淄博汇能环保设备有限公司 | 【铜排折弯机,钢丝折弯成型机,汽车发泡钢丝折弯机,线材折弯机厂家,线材成型机,铁线折弯机】贝朗折弯机厂家_东莞市贝朗自动化设备有限公司 | 冷却塔风机厂家_静音冷却塔风机_冷却塔电机维修更换维修-广东特菱节能空调设备有限公司 | 温州中研白癜风专科_温州治疗白癜风_温州治疗白癜风医院哪家好_温州哪里治疗白癜风 | 不锈钢复合板厂家_钛钢复合板批发_铜铝复合板供应-威海泓方金属复合材料股份有限公司 | 精密机械零件加工_CNC加工_精密加工_数控车床加工_精密机械加工_机械零部件加工厂 | 涂层测厚仪_漆膜仪_光学透过率仪_十大创新厂家-果欧电子科技公司 | uv机-uv灯-uvled光固化机-生产厂家-蓝盾机电 | 吉林污水处理公司,长春工业污水处理设备,净水设备-长春易洁环保科技有限公司 | 茅茅虫AI论文写作助手-免费AIGC论文查重_写毕业论文降重 | 微型实验室真空泵-无油干式真空泵-微型涡旋耐腐蚀压缩机-思科涡旋科技(杭州)有限公司 | 净化车间_洁净厂房_净化公司_净化厂房_无尘室工程_洁净工程装修|改造|施工-深圳净化公司 | 皮带式输送机械|链板式输送机|不锈钢输送机|网带输送机械设备——青岛鸿儒机械有限公司 | 百方网-百方电气网,电工电气行业专业的B2B电子商务平台 | 管形母线,全绝缘铜管母线厂家-山东佰特电气科技有限公司 | 上海冠顶工业设备有限公司-隧道炉,烘箱,UV固化机,涂装设备,高温炉,工业机器人生产厂家 | 渗透仪-直剪仪-三轴仪|苏州昱创百科 |