網站首頁 編程語言 正文
CPython 中的垃圾收集器
CPython 的垃圾收集器(簡稱GC)是 Python 內置的為了解決循環引用問題的方法。默認情況下,它總是在后臺運行,并且每隔一段時間就會發揮它的魔力,所以你不必擔心循環引用物會堵塞你的內存。
垃圾收集器被設計為從 CPython 的工作內存中找到并刪除循環引用對象。它通過以下方式完成這一工作。
- 檢測循環引用的對象
- 調用最終的
__del__
方法 - 它從每個對象中刪除指針(以此來解決循環問題),只有當循環在步驟 2 之后仍然是孤立的
在這個過程完成后,以前在循環中的每個對象現在的引用計數都是 0 ,因此此對象將從內存中刪除。
雖然它是自動工作的,但實際上我們可以把它作為一個模塊從標準庫中導入。舉例如下:
import gc
檢測循環引用
CPython 的垃圾收集器會跟蹤內存中存在的各種對象--但不是所有的對象。我們可以實例化一些對象,看看垃圾收集器是否會收集它們。
>>> gc.is_tracked("a string") False >>> gc.is_tracked(["a", "list"]) True
如果一個對象可以包含指針,這就使它有能力形成循環引用結構的一部分--而這正是垃圾檢測器存在的目的,即檢測和拆除。在 Python 中這樣的對象通常被稱為 "容器對象"。
所以,垃圾收集器需要知道任何有可能作為循環引用的一部分而存在的對象。字符串不能,所以 "一個字符串 "不會被垃圾收集器追蹤。列表(正如我們已經看到的)能夠包含指針,因此 ['a', 'list']
被跟蹤。
用戶定義的類的任何實例也將被垃圾收集器跟蹤,因為我們總是可以在它們身上設置任意的屬性(指針)。
>>> Wade = MyNameClass("Wade") >>> gc.is_tracked(Wade) True
所以,垃圾收集器知道所有有可能形成循環引用的對象。它怎么知道是否已經形成循環引用呢?
它也知道每個對象中的所有指針,以及它們所指向的位置。我們可以看到這個動作。
>>> my_list = ["a", "list"] >>> gc.get_referents(my_list) ['list', 'a']
get_referents
方法(也稱為遍歷方法)接收一個對象,并返回它所包含的對象指針的列表(它的引用)。因此,上面的列表包含指向其每個元素的指針,這些元素都是字符串。
讓我們在一個對象的循環中看看 get_referents
方法(雖然還不是一個循環引用,因為這些對象仍然可以從命名空間中被訪問)。
>>> jane = MyNamedClass("Jane") >>> bob = MyNamedClass("Bob") >>> jane.friend = bob >>> bob.friend = jane >>> gc.get_referents(bob) [{'name': 'bob', 'friend': <__main__.MyNamedClass object at 0x7ff29a095d60>}, <class '__main__
在這個循環中,我們可以看到由 bob 指向的對象包含指向以下內容的指針:它的屬性字典,包含 bob 的名字 (bob) 和它的朋友 (同樣由 jane 指向的 MyNamedClass 實例) 。bob 對象也有一個指向類對象本身的指針,因為 bob.class
將返回那個類對象。
當垃圾收集器運行時,它檢查它所知道的每個對象(也就是當你調用 gc.is_tracked
時返回True的任何對象)是否可以從命名空間到達。它通過跟蹤來自命名空間的所有指針,以及這些指針所指向的對象中的指針,以此類推,直到它建立起所有可從代碼中訪問的東西的整個視圖。
如果在做完這些之后,GC 發現存在一些不能從命名空間到達的對象,那么它可以把這些對象清除掉。
記住,任何仍在內存中的對象必須有一個非零的引用計數,否則它們會因為引用計數而被刪除。對于那些無法到達但仍有非零引用計數的對象,它們必須是循環引用的一部分,這就是為什么我們如此關心這些發生的可能性。
讓我們回到引用循環,jane 和 bob,通過從命名空間中移除指針,把這個循環變成一個循環的隔離。
>>> del jane >>> del bob
現在,我們已經了解了垃圾收集器所要解決的確切情況。我們可以通過調用 gc.collect()
來觸發手動垃圾收集。
>>> gc.collect() Deleting Bob! Deleting Jane! 4
默認情況下,垃圾收集器會每隔一段時間自動執行這個動作(因為越來越多的對象在CPython運行時被創建和銷毀)。
在上面的代碼片段中,我們看到的輸出包含了來自 MyNamClass 的 __del__
方法的打印語句,在最后有一個數字--在這個例子中,是 4。 這個數字是由垃圾收集器本身輸出的,它告訴我們有多少對象被移除。
參考鏈接:
https://anvil.works/articles/pointers-in-my-python-3
原文鏈接:https://juejin.cn/post/7151052035876618254
相關推薦
- 2021-12-04 C#獲取Windows10屏幕縮放比例的操作方法_C#教程
- 2022-05-05 Python學習之列表常用方法總結_python
- 2022-09-25 ubuntu 20.04 開機黑屏小橫杠和循環登錄
- 2022-05-13 深度優先搜索之八皇后問題
- 2022-07-07 Python數據分析之?Matplotlib?3D圖詳情_python
- 2022-05-08 SQL利用游標遍歷日期查詢的過程詳解_MsSql
- 2022-10-19 Android開發flow常見API的使用示例詳解_Android
- 2023-01-15 C#實現虛擬鍵盤的實例詳解_C#教程
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支