<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          Python垃圾回收

          共 2458字,需瀏覽 5分鐘

           ·

          2022-01-09 06:39


          在計(jì)算機(jī)科學(xué)中,GC 是一種自動(dòng)的內(nèi)存管理機(jī)制。當(dāng)對(duì)應(yīng)內(nèi)存不再需要的時(shí)候,就應(yīng)該予以釋放,這種內(nèi)存資源管理,稱(chēng)為垃圾回收。而且垃圾回收器會(huì)自行進(jìn)行垃圾對(duì)象的處理,可以讓程序員減少很多負(fù)擔(dān),也減少了程序員犯錯(cuò)誤的機(jī)會(huì)。


          • 垃圾回收

          我們作為Python程序員也是非常幸福的,我們?nèi)粘2惶枰P(guān)注內(nèi)存管理和垃圾回收,是因?yàn)?/span>CPython的解釋器有一套自己的機(jī)制來(lái)處理。那么,在Python的世界里為什么不太需要關(guān)注垃圾回收呢?

          這是因?yàn)?/span>Python自己的解釋器自動(dòng)做了垃圾回收相應(yīng)的處理,在絕大部分場(chǎng)景下是不需要人為的干涉的。另外,大家對(duì)于Python的共識(shí)就是開(kāi)發(fā)效率。因?yàn)槠涫且粋€(gè)膠水語(yǔ)言,在很多場(chǎng)景下高性能以及內(nèi)存問(wèn)題其實(shí)并不凸顯,而且現(xiàn)在服務(wù)器資源很便宜而人力資源很貴的情況下。

          使用PythonWeb開(kāi)發(fā),工作很多年也不太會(huì)遇到內(nèi)存管理和垃圾回收的。在Web應(yīng)用幾乎都是使用多進(jìn)程模型的,一則是會(huì)有定期超時(shí)重啟的機(jī)制,二是每次上線的操作也會(huì)進(jìn)程的重啟。所以不會(huì)有某個(gè)進(jìn)程長(zhǎng)時(shí)間的駐留,使其占用很多內(nèi)存,導(dǎo)致內(nèi)存泄漏。所以,GC的缺陷基本不太會(huì)對(duì)Web開(kāi)發(fā)產(chǎn)生很大的影響。且CPython也足夠完善,基本不太會(huì)出現(xiàn)內(nèi)存泄漏這樣的問(wèn)題。大部分場(chǎng)景下,都是因?yàn)殚_(kāi)發(fā)者錯(cuò)誤的使用或者是誤判導(dǎo)致內(nèi)存占用不正常。

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

          Python的垃圾回收是建立在引用技術(shù)上的,所以理解引用計(jì)數(shù)也是非常重要的。而引用計(jì)數(shù)的原理就是,當(dāng)一個(gè)對(duì)象的引用被創(chuàng)建或者復(fù)制時(shí),對(duì)象的引用計(jì)數(shù)加1;當(dāng)一個(gè)對(duì)象的引用被銷(xiāo)毀時(shí),對(duì)象的引用計(jì)數(shù)減1;當(dāng)對(duì)象的引用計(jì)數(shù)減少為0時(shí),就意味著對(duì)象已經(jīng)沒(méi)有被任何人使用了,可以將其所占用的內(nèi)存立刻釋放了。

          引用計(jì)數(shù)這種機(jī)制的特點(diǎn)是,有比較好的實(shí)時(shí)性,但是引用計(jì)數(shù)會(huì)有一個(gè)循環(huán)引用的問(wèn)題。比如說(shuō)A引用了B,而B又引用了A,導(dǎo)致每一個(gè)對(duì)象的引用計(jì)數(shù)都不為0,那么AB占用的內(nèi)存資源永遠(yuǎn)都不會(huì)被回收。所以,就需要一些回收算法來(lái)解決這個(gè)問(wèn)題,而Python就是使用了標(biāo)記清除分代回收機(jī)制。

          sys.getrefcount()
          • 標(biāo)記清除

          上面我們說(shuō)了,標(biāo)記-清除就是為解決循環(huán)引用的問(wèn)題。最理想的情況下,比如說(shuō)有兩個(gè)對(duì)象AB,其中A有一個(gè)B的引用,就會(huì)將B的引用計(jì)數(shù)減1。然后順著引用達(dá)到B,因?yàn)?/span>B有一個(gè)引用了A,同樣將A的引用計(jì)數(shù)減少1。這樣,就將引用計(jì)數(shù)中循環(huán)引用的環(huán)給摘除。
          但是,還會(huì)存在另外一個(gè)問(wèn)題。假設(shè)對(duì)象A,它有一個(gè)對(duì)象C的引用,而C并沒(méi)有引用A。如果將C的引用計(jì)數(shù)減少1,而最后A沒(méi)有被回收,顯然我們錯(cuò)誤將C的引用計(jì)數(shù)減少了1。這樣,將導(dǎo)致在未來(lái)的某個(gè)時(shí)段出現(xiàn)了一個(gè)對(duì)C的懸空引用。這就要求我們?cè)?/span>C沒(méi)有被刪除的情況下,復(fù)用C的引用計(jì)數(shù)。如果采用這樣方案的話,那么維護(hù)這個(gè)引用計(jì)數(shù)的復(fù)雜度就會(huì)成倍的增加。而這個(gè)標(biāo)記清除采用了更好的做法來(lái)解決這個(gè)問(wèn)題。
          標(biāo)記清除采用了更好的做法,它并不改動(dòng)真實(shí)的引用計(jì)數(shù),而是將集合中對(duì)象的引用計(jì)數(shù)復(fù)制一份副本,改動(dòng)該對(duì)象引用的副本。對(duì)于副本做任何的改動(dòng),都不會(huì)影響到對(duì)象生命周期的維護(hù)。
          • 分代回收
          分代回收是在面試中,常常會(huì)被問(wèn)到的一個(gè)問(wèn)題。分代回收的核心思想就是,對(duì)象存活的時(shí)間越長(zhǎng),越不可能是垃圾,應(yīng)該更少的去回收。且Python將所有的對(duì)象分為012三代,所有的新建對(duì)象都是0代對(duì)象。但是,當(dāng)某一代對(duì)象經(jīng)歷過(guò)垃圾回收,依然存活,那么它就被歸入下一代對(duì)象,即1代或者2代了。
          分代回收的預(yù)值,可以使用如下代碼進(jìn)行查看。通常,返回一個(gè)元組且包含三個(gè)數(shù)值,默認(rèn)值為(700, 10, 10)。其中第一個(gè)數(shù)值700表示,從上一個(gè)垃圾回收到現(xiàn)在分配內(nèi)存的數(shù)目減去釋放內(nèi)存的數(shù)目。如果這個(gè)數(shù)值到了700,則會(huì)對(duì)第一代的垃圾對(duì)象進(jìn)行回收,并且給第二個(gè)數(shù)值加1。當(dāng)?shù)诙€(gè)數(shù)值增加到10的時(shí)候,就會(huì)對(duì)第一代和第二代的垃圾對(duì)象進(jìn)行回收,并且給第三個(gè)數(shù)值加1。當(dāng)?shù)谌齻€(gè)數(shù)值增加到10的時(shí)候,則三代都會(huì)被回收,然后初始化為(0, 0, 0)并繼續(xù)開(kāi)始計(jì)數(shù)。
          需要注意的是,如果沒(méi)有十分必要的場(chǎng)景,這個(gè)分代回收的默認(rèn)值通常是不需要我們?nèi)藶榈母膭?dòng)的。
          In [1]: import gc
          In [2]: gc.get_threshold()Out[2]: (700, 10, 10)
          • 強(qiáng)制回收
          上面介紹了Python的自動(dòng)垃圾回收機(jī)制,而Python也支持在某一刻特定的時(shí)間點(diǎn),使用gc.collect()方法強(qiáng)制回收。不會(huì),通常我們是不適用強(qiáng)制回收的,而是使用下面這種禁用垃圾回收的方式。
          • 禁用垃圾回收
          這個(gè)垃圾回收機(jī)制不是挺好的,那我們會(huì)什么還要禁用呢。通常我們禁用GC的一個(gè)場(chǎng)景就是,某一段代碼中需要加載大量的原始數(shù)據(jù),尤其是有大量的新建、刪除對(duì)象這樣的操作。也就是執(zhí)行某一段代碼的時(shí)候,會(huì)自動(dòng)觸發(fā)很多次的垃圾回收。但是,我們需要知道Python執(zhí)行垃圾回收的時(shí)候,它會(huì)暫停當(dāng)前的工作。所以,這種工作耗時(shí)越多就會(huì)拖累我們程序的運(yùn)行時(shí)間。
          那我們?cè)趺崔k呢?我們通常都會(huì)在執(zhí)行這段代碼之前,禁用垃圾回收,執(zhí)行完之后再手動(dòng)開(kāi)啟。熟悉開(kāi)源項(xiàng)目的同學(xué)可以會(huì)看到,有些項(xiàng)目中會(huì)使用gc.set_threshold(0)而不用gc.disable這種寫(xiě)法。是因?yàn)橛行┑谌降膸?kù)會(huì)隱式的啟用GCgc.disable不起作用了,而使用gc.set_threshold(0)就不會(huì)有第三方的庫(kù)把垃圾回收開(kāi)啟了,除非我們想要把它開(kāi)啟。
          gc.disable()do somethingsgc.enable()

          原文鏈接:https://www.escapelife.site/posts/57aa800d.html

          文章轉(zhuǎn)載:Python編程學(xué)習(xí)圈
          (版權(quán)歸原作者所有,侵刪)

          點(diǎn)擊下方“閱讀原文”查看更多

          瀏覽 55
          點(diǎn)贊
          評(píng)論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          評(píng)論
          圖片
          表情
          推薦
          點(diǎn)贊
          評(píng)論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  大香蕉网|伊人在线大香蕉|大香蕉|大香蕉网站 | 亚洲91成人电影 | 小早川怜子爆乿护士中文 | 国产a不卡 | 大香蕉在线视频99 |