www.久久久久|狼友网站av天堂|精品国产无码a片|一级av色欲av|91在线播放视频|亚洲无码主播在线|国产精品草久在线|明星AV网站在线|污污内射久久一区|婷婷综合视频网站

當(dāng)前位置:首頁(yè) > 單片機(jī) > 程序員小灰
[導(dǎo)讀]什么是內(nèi)存管理器(what) Python作為一個(gè)高層次的結(jié)合了解釋性、編譯性、互動(dòng)性和面向?qū)ο蟮哪_本語(yǔ)言,與大多數(shù)編程語(yǔ)言不同,Python中的變量無(wú)需事先申明,變量無(wú)需指定類型,程序員無(wú)需關(guān)心內(nèi)存管理,Python解釋器給你自動(dòng)回收。開(kāi)發(fā)人員不用過(guò)多的關(guān)心內(nèi)

什么是內(nèi)存管理器(what)

Python作為一個(gè)高層次的結(jié)合了解釋性、編譯性、互動(dòng)性和面向?qū)ο蟮哪_本語(yǔ)言,與大多數(shù)編程語(yǔ)言不同,Python中的變量無(wú)需事先申明,變量無(wú)需指定類型,程序員無(wú)需關(guān)心內(nèi)存管理,Python解釋器給你自動(dòng)回收。開(kāi)發(fā)人員不用過(guò)多的關(guān)心內(nèi)存管理機(jī)制,這一切全部由python內(nèi)存管理器承擔(dān)了復(fù)雜的內(nèi)存管理工作。

內(nèi)存不外乎創(chuàng)建和銷(xiāo)毀兩部分,本文將圍繞python的內(nèi)存池和垃圾回收兩部分進(jìn)行分析。

Python內(nèi)存池

為什么要引入內(nèi)存池(why)

當(dāng)創(chuàng)建大量消耗小內(nèi)存的對(duì)象時(shí),頻繁調(diào)用new/malloc會(huì)導(dǎo)致大量的內(nèi)存碎片,致使效率降低。內(nèi)存池的作用就是預(yù)先在內(nèi)存中申請(qǐng)一定數(shù)量的,大小相等的內(nèi)存塊留作備用,當(dāng)有新的內(nèi)存需求時(shí),就先從內(nèi)存池中分配內(nèi)存給這個(gè)需求,不夠之后再申請(qǐng)新的內(nèi)存。這樣做最顯著的優(yōu)勢(shì)就是能夠減少內(nèi)存碎片,提升效率。

python中的內(nèi)存管理機(jī)制為Pymalloc

內(nèi)存池是如何工作的(how)

首先,我們看一張CPython(python解釋器)的內(nèi)存架構(gòu)圖:

  • python的對(duì)象管理主要位于Level+1~Level+3層

  • Level+3層:對(duì)于python內(nèi)置的對(duì)象(比如int,dict等)都有獨(dú)立的私有內(nèi)存池,對(duì)象之間的內(nèi)存池不共享,即int釋放的內(nèi)存,不會(huì)被分配給float使用

  • Level+2層:當(dāng)申請(qǐng)的內(nèi)存大小小于256KB時(shí),內(nèi)存分配主要由 Python 對(duì)象分配器(Python’s object allocator)實(shí)施

  • Level+1層:當(dāng)申請(qǐng)的內(nèi)存大小大于256KB時(shí),由Python原生的內(nèi)存分配器進(jìn)行分配,本質(zhì)上是調(diào)用C標(biāo)準(zhǔn)庫(kù)中的malloc/realloc等函數(shù)

關(guān)于釋放內(nèi)存方面,當(dāng)一個(gè)對(duì)象的引用計(jì)數(shù)變?yōu)?時(shí),Python就會(huì)調(diào)用它的析構(gòu)函數(shù)。調(diào)用析構(gòu)函數(shù)并不意味著最終一定會(huì)調(diào)用free來(lái)釋放內(nèi)存空間,如果真是這樣的話,那頻繁地申請(qǐng)、釋放內(nèi)存空間會(huì)使Python的執(zhí)行效率大打折扣。因此在析構(gòu)時(shí)也采用了內(nèi)存池機(jī)制,從內(nèi)存池申請(qǐng)到的內(nèi)存會(huì)被歸還到內(nèi)存池中,以避免頻繁地申請(qǐng)和釋放動(dòng)作。

垃圾回收機(jī)制

Python的垃圾回收機(jī)制采用引用計(jì)數(shù)機(jī)制為主,標(biāo)記-清除和分代回收機(jī)制為輔的策略。其中,標(biāo)記-清除機(jī)制用來(lái)解決計(jì)數(shù)引用帶來(lái)的循環(huán)引用而無(wú)法釋放內(nèi)存的問(wèn)題,分代回收機(jī)制是為提升垃圾回收的效率。

引用計(jì)數(shù)

Python通過(guò)引用計(jì)數(shù)來(lái)保存內(nèi)存中的變量追蹤,即記錄該對(duì)象被其他使用的對(duì)象引用的次數(shù)。

Python中有個(gè)內(nèi)部跟蹤變量叫做引用計(jì)數(shù)器,每個(gè)變量有多少個(gè)引用,簡(jiǎn)稱引用計(jì)數(shù)。當(dāng)某個(gè)對(duì)象的引用計(jì)數(shù)為0時(shí),就列入了垃圾回收隊(duì)列。

   
  1. >>> a=[1,2]

  2. >>> import sys

  3. >>> sys.getrefcount(a) ## 獲取對(duì)象a的引用次數(shù)

  4. 2

  5. >>> b=a

  6. >>> sys.getrefcount(a)

  7. 3

  8. >>> del b ## 刪除b的引用

  9. >>> sys.getrefcount(a)

  10. 2

  11. >>> c=list()

  12. >>> c.append(a) ## 加入到容器中

  13. >>> sys.getrefcount(a)

  14. 3

  15. >>> del c ## 刪除容器,引用-1

  16. >>> sys.getrefcount(a)

  17. 2

  18. >>> b=a

  19. >>> sys.getrefcount(a)

  20. 3

  21. >>> a=[3,4] ## 重新賦值

  22. >>> sys.getrefcount(a)

  23. 2

注意:當(dāng)把a(bǔ)作為參數(shù)傳遞給getrefcount時(shí),會(huì)產(chǎn)生一個(gè)臨時(shí)的引用,因此得出來(lái)的結(jié)果比真實(shí)情況+1

  • 引用計(jì)數(shù)增加的情況:

  1. 一個(gè)對(duì)象被分配給一個(gè)新的名字(例如:a=[1,2])

  2. 將其放入一個(gè)容器中(如列表、元組或字典)(例如:c.append(a))

  • 引用計(jì)數(shù)減少的情況:

  1. 使用del語(yǔ)句對(duì)對(duì)象別名顯式的銷(xiāo)毀(例如:del b)

  2. 對(duì)象所在的容器被銷(xiāo)毀或從容器中刪除對(duì)象(例如:del c )

  3. 引用超出作用域或被重新賦值(例如:a=[3,4])

引用計(jì)數(shù)能夠解決大多數(shù)垃圾回收的問(wèn)題,但是遇到兩個(gè)對(duì)象相互引用的情況,del語(yǔ)句可以減少引用次數(shù),但是引用計(jì)數(shù)不會(huì)歸0,對(duì)象也就不會(huì)被銷(xiāo)毀,從而造成了內(nèi)存泄漏問(wèn)題。針對(duì)該情況,Python引入了標(biāo)記-清除機(jī)制。

標(biāo)記-清除

標(biāo)記-清除用來(lái)解決引用計(jì)數(shù)機(jī)制產(chǎn)生的循環(huán)引用,進(jìn)而導(dǎo)致內(nèi)存泄漏的問(wèn)題 。循環(huán)引用只有在容器對(duì)象才會(huì)產(chǎn)生,比如字典,元組,列表等。

顧名思義,該機(jī)制在進(jìn)行垃圾回收時(shí)分成了兩步,分別是:

  • 標(biāo)記階段,遍歷所有的對(duì)象,如果是可達(dá)的(reachable),也就是還有對(duì)象引用它,那么就標(biāo)記該對(duì)象為可達(dá)

  • 清除階段,再次遍歷對(duì)象,如果發(fā)現(xiàn)某個(gè)對(duì)象沒(méi)有標(biāo)記為可達(dá)(即為Unreachable),則就將其回收

   
  1. >>> a=[1,2]

  2. >>> b=[3,4]

  3. >>> sys.getrefcount(a)

  4. 2

  5. >>> sys.getrefcount(b)

  6. 2

  7. >>> a.append(b)

  8. >>> sys.getrefcount(b)

  9. 3

  10. >>> b.append(a)

  11. >>> sys.getrefcount(a)

  12. 3

  13. >>> del a

  14. >>> del b

  • a引用b,b引用a,此時(shí)兩個(gè)對(duì)象各自被引用了2次(去除getrefcout()的臨時(shí)引用)

  • 執(zhí)行del之后,對(duì)象a,b的引用次數(shù)都-1,此時(shí)各自的引用計(jì)數(shù)器都為1,陷入循環(huán)引用

  • 標(biāo)記:找到其中的一端a,因?yàn)樗幸粋€(gè)對(duì)b的引用,則將b的引用計(jì)數(shù)-1

  • 標(biāo)記:再沿著引用到b,b有一個(gè)a的引用,將a的引用計(jì)數(shù)-1,此時(shí)對(duì)象a和b的引用次數(shù)全部為0,被標(biāo)記為不可達(dá)(Unreachable)

  • 清除: 被標(biāo)記為不可達(dá)的對(duì)象就是真正需要被釋放的對(duì)象

上面描述的垃圾回收的階段,會(huì)暫停整個(gè)應(yīng)用程序,等待標(biāo)記清除結(jié)束后才會(huì)恢復(fù)應(yīng)用程序的運(yùn)行。為了減少應(yīng)用程序暫停的時(shí)間,Python 通過(guò)“分代回收”(Generational Collection)以空間換時(shí)間的方法提高垃圾回收效率。

分代回收

分代回收是基于這樣的一個(gè)統(tǒng)計(jì)事實(shí),對(duì)于程序,存在一定比例的內(nèi)存塊的生存周期比較短;而剩下的內(nèi)存塊,生存周期會(huì)比較長(zhǎng),甚至?xí)某绦蜷_(kāi)始一直持續(xù)到程序結(jié)束。生存期較短對(duì)象的比例通常在 80%~90%之間。因此,簡(jiǎn)單地認(rèn)為:對(duì)象存在時(shí)間越長(zhǎng),越可能不是垃圾,應(yīng)該越少去收集。這樣在執(zhí)行標(biāo)記-清除算法時(shí)可以有效減小遍歷的對(duì)象數(shù),從而提高垃圾回收的速度,是一種以空間換時(shí)間的方法策略

Python將所有的對(duì)象分為年輕代(第0代)、中年代(第1代)、老年代(第2代)三代。所有的新建對(duì)象默認(rèn)是 第0代對(duì)象。當(dāng)在第0代的gc掃描中存活下來(lái)的對(duì)象將被移至第1代,在第1代的gc掃描中存活下來(lái)的對(duì)象將被移至第2代。

gc掃描次數(shù)(第0代>第1代>第2代)

當(dāng)某一代中被分配的對(duì)象與被釋放的對(duì)象之差達(dá)到某一閾值時(shí),就會(huì)觸發(fā)當(dāng)前一代的gc掃描。當(dāng)某一代被掃描時(shí),比它年輕的一代也會(huì)被掃描,因此,第2代的gc掃描發(fā)生時(shí),第0,1代的gc掃描也會(huì)發(fā)生,即為全代掃描。

   
  1. >>> import gc

  2. >>> gc.get_threshold() ## 分代回收機(jī)制的參數(shù)閾值設(shè)置

  3. (700, 10, 10)

  • 700=新分配的對(duì)象數(shù)量-釋放的對(duì)象數(shù)量,第0代gc掃描被觸發(fā)

  • 第一個(gè)10:第0代gc掃描發(fā)生10次,則第1代的gc掃描被觸發(fā)

  • 第二個(gè)10:第1代的gc掃描發(fā)生10次,則第2代的gc掃描被觸發(fā)

思考

在標(biāo)記-清除中,如果對(duì)象c也引用a,執(zhí)行del操作后,會(huì)發(fā)生什么?

對(duì)象a,b,c的引用關(guān)系如下圖所示:

   
  1. >>> a=[1,2]

  2. >>> b=[3,4]

  3. >>> c=a

  4. >>> a.append(b)

  5. >>> b.append(a)

  • ref_count表示引用計(jì)數(shù)

  • 對(duì)象a,b,c全部為reachable

執(zhí)行del之后,引用關(guān)系如下圖所示:

   
  1. >>> del a

  2. >>> del b

  • a,b,c的ref_count減1

執(zhí)行g(shù)c掃描

  • 標(biāo)記: a引用b,將b的refcount減1到0,b引用a,將a的refcount減1到1,將b放在unreachable下 

  • 再循環(huán):因?yàn)閍是可達(dá)的,所以會(huì)遞歸地將從a節(jié)點(diǎn)出發(fā)可以達(dá)到的所有節(jié)點(diǎn)標(biāo)記為reachable下,即為: 

  • 清除:unreachable下沒(méi)有可清除的對(duì)象,因此a,b,c對(duì)象不會(huì)被清除

總結(jié)

總體而言,python通過(guò)內(nèi)存池來(lái)減少內(nèi)存碎片化,提高執(zhí)行效率。主要通過(guò)引用計(jì)數(shù)來(lái)完成垃圾回收,通過(guò)標(biāo)記-清除解決容器對(duì)象循環(huán)引用造成的問(wèn)題,通過(guò)分代回收提高垃圾回收的效率。


—————END—————



喜歡本文的朋友,歡迎關(guān)注公眾號(hào) 程序員小灰,收看更多精彩內(nèi)容

      
點(diǎn)個(gè)[在看],是對(duì)小灰最大的支持!


免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺(tái)僅提供信息存儲(chǔ)服務(wù)。文章僅代表作者個(gè)人觀點(diǎn),不代表本平臺(tái)立場(chǎng),如有問(wèn)題,請(qǐng)聯(lián)系我們,謝謝!

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: 驅(qū)動(dòng)電源

在工業(yè)自動(dòng)化蓬勃發(fā)展的當(dāng)下,工業(yè)電機(jī)作為核心動(dòng)力設(shè)備,其驅(qū)動(dòng)電源的性能直接關(guān)系到整個(gè)系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動(dòng)勢(shì)抑制與過(guò)流保護(hù)是驅(qū)動(dòng)電源設(shè)計(jì)中至關(guān)重要的兩個(gè)環(huán)節(jié),集成化方案的設(shè)計(jì)成為提升電機(jī)驅(qū)動(dòng)性能的關(guān)鍵。

關(guān)鍵字: 工業(yè)電機(jī) 驅(qū)動(dòng)電源

LED 驅(qū)動(dòng)電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個(gè)照明設(shè)備的使用壽命。然而,在實(shí)際應(yīng)用中,LED 驅(qū)動(dòng)電源易損壞的問(wèn)題卻十分常見(jiàn),不僅增加了維護(hù)成本,還影響了用戶體驗(yàn)。要解決這一問(wèn)題,需從設(shè)計(jì)、生...

關(guān)鍵字: 驅(qū)動(dòng)電源 照明系統(tǒng) 散熱

根據(jù)LED驅(qū)動(dòng)電源的公式,電感內(nèi)電流波動(dòng)大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關(guān)鍵字: LED 設(shè)計(jì) 驅(qū)動(dòng)電源

電動(dòng)汽車(chē)(EV)作為新能源汽車(chē)的重要代表,正逐漸成為全球汽車(chē)產(chǎn)業(yè)的重要發(fā)展方向。電動(dòng)汽車(chē)的核心技術(shù)之一是電機(jī)驅(qū)動(dòng)控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機(jī)驅(qū)動(dòng)系統(tǒng)中的關(guān)鍵元件,其性能直接影響到電動(dòng)汽車(chē)的動(dòng)力性能和...

關(guān)鍵字: 電動(dòng)汽車(chē) 新能源 驅(qū)動(dòng)電源

在現(xiàn)代城市建設(shè)中,街道及停車(chē)場(chǎng)照明作為基礎(chǔ)設(shè)施的重要組成部分,其質(zhì)量和效率直接關(guān)系到城市的公共安全、居民生活質(zhì)量和能源利用效率。隨著科技的進(jìn)步,高亮度白光發(fā)光二極管(LED)因其獨(dú)特的優(yōu)勢(shì)逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

關(guān)鍵字: 發(fā)光二極管 驅(qū)動(dòng)電源 LED

LED通用照明設(shè)計(jì)工程師會(huì)遇到許多挑戰(zhàn),如功率密度、功率因數(shù)校正(PFC)、空間受限和可靠性等。

關(guān)鍵字: LED 驅(qū)動(dòng)電源 功率因數(shù)校正

在LED照明技術(shù)日益普及的今天,LED驅(qū)動(dòng)電源的電磁干擾(EMI)問(wèn)題成為了一個(gè)不可忽視的挑戰(zhàn)。電磁干擾不僅會(huì)影響LED燈具的正常工作,還可能對(duì)周?chē)娮釉O(shè)備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來(lái)解決L...

關(guān)鍵字: LED照明技術(shù) 電磁干擾 驅(qū)動(dòng)電源

開(kāi)關(guān)電源具有效率高的特性,而且開(kāi)關(guān)電源的變壓器體積比串聯(lián)穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機(jī)重量也有所下降,所以,現(xiàn)在的LED驅(qū)動(dòng)電源

關(guān)鍵字: LED 驅(qū)動(dòng)電源 開(kāi)關(guān)電源

LED驅(qū)動(dòng)電源是把電源供應(yīng)轉(zhuǎn)換為特定的電壓電流以驅(qū)動(dòng)LED發(fā)光的電壓轉(zhuǎn)換器,通常情況下:LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: LED 隧道燈 驅(qū)動(dòng)電源
關(guān)閉