Python實現模塊熱載入

来源:https://www.cnblogs.com/kanadeblisst/archive/2023/12/18/17912027.html
-Advertisement-
Play Games

為什麼需要熱載入 在某些情況,你可能不希望關閉Python進程並重新打開,或者你無法重新啟動Python,這時候就需要實現實時修改代碼實時生效,而不用重新啟動Python 在我的需求下,這個功能非常重要,我將Python註入到了其他進程,並作為一個線程運行。如果我想關閉Python,要麼殺死Pyth ...


為什麼需要熱載入

在某些情況,你可能不希望關閉Python進程並重新打開,或者你無法重新啟動Python,這時候就需要實現實時修改代碼實時生效,而不用重新啟動Python

在我的需求下,這個功能非常重要,我將Python註入到了其他進程,並作為一個線程運行。如果我想關閉Python,要麼殺死Python相關的線程,要麼重新啟動進程,這都比較麻煩。所以當我修改完代碼後,熱載入代碼是最方便的方法

Python中的導入機制

我們重覆導入一個庫時,第二次導入時並沒有運行庫裡面的代碼,比如先寫一個a.py,在裡面寫一行代碼print("a模塊載入"),然後在寫一個b.py, 裡面寫兩行import a。即使你在多線程中再導入一遍a模塊,也不會列印。例如下麵的代碼:

import a
import threading
print(id(a))

def test():
    import a
    print(id(a))

threading.Thread(target=test).start()

可以看到a的id是一樣的,也就是同一個對象。

為什麼會這樣呢?這和Python的模塊導入機制有關,Python會在sys.modules這個字典里存儲著所有的全局模塊,當你導入一個新模塊時,他會先查找sys.modules里有沒有這個模塊,如果沒有再導入,如果有就在當前代碼增加個引用。舉個最簡單的例子:

a.py

print("a模塊載入")

def aa():
    print("a模塊中的aa方法被載入")

b.py

import sys
a = sys.modules["a"]
a.aa()

c.py

import a
import b

先導入a模塊,這樣sys.modules已經有了a模塊,你就可以使用sys.modules["a"]來使用a模塊,它和import a基本是一樣的。如果你先import b就會發現sys.modules不存在a

重新導入模塊1

既然知道它是先查找sys.modules,那我在導入之前,先刪除掉裡面的a再導入就可以了

import a
import sys
del sys.modules["a"]
import a

這樣就能重新載入模塊

重新導入模塊2

Python基礎庫也提供了一個方法重新載入模塊:

import a
import importlib

importlib.reload(a)

看一下內部代碼是怎麼實現的:

邏輯也比較簡單, 先看sys.modules里有沒有這個模塊,如果有就使用_bootstrap._exec導入模塊。我們是不是也可以通過_bootstrap._exec來重新導入模塊,可以但不建議,因為下劃線開頭的模塊或者函數都是不建議外部使用的,這些介面可能在版本更新後變動比較頻繁

無法熱載入的情況

__main__模塊無法熱載入。當你執行python a.py,這個a.py文件是無法熱載入的,它並沒有作為模塊導入,在sys.modules的名稱就是__main__

如果你在__main__使用from a import A導入的類,即使a模塊重新載入,__main__裡面的A也不會改變

熱載入無法影響已經實例化的對象,比如你修改了模塊裡面的類代碼,但是已經在__main__里實例化了這個類對象,並且一直使用未釋放,它的邏輯在熱載入之後不會受影響。

函數級熱載入

要想實現函數、方法乃至對象級別的熱載入,得修改記憶體中的Python對象。有一個項目實現了這種,有興趣的可以看:https://github.com/breuleux/jurigged

我的需求沒有這麼細,就不測試了

監聽文件變化

我選擇的是watchdog,另一個pyinotify不支持Windows。

watchdog在Windows上有點小bug,修改文件會觸發兩次事件。搜到一個解決方案:不使用預設的事件觸發,而是利用文件快照,每隔一段時間做一次比對。原文鏈接:Python神器watchdog(監控文件變化),我測試了一下效果很好。

源碼

完整的源碼就不放了,具體可以看:https://github.com/kanadeblisst00/module_hot_loading

國內倉庫:http://www.pygrower.cn:21180/kanadeblisst/module_hot_loading

安裝

pip install module-hot-loading

使用

from threading import Event
from module_hot_loading import monitor_dir


if __name__ == "__main__":
    event = Event()
    event.set()
    path = "."
    monitor_dir(path, event, __file__, interval=2, only_import_exist=False)
    

monitor_dir的參數:

  1. 需要監控的目錄路徑
  2. 停止監控的事件信號
  3. __main__的代碼文件路徑
  4. interval: 每隔幾秒打一次文件快照做比對
  5. only_import_exist: 只重新載入已經導入的模塊

效果


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 工欲善其事,必先利其器。Chrome 可能是前端開發中使用最多的瀏覽器。在日常開發中,下列幾款 Chrome 擴展也許能讓你的開發工作事半功倍
  • 大家好,我是 Java陳序員。 這幾天,逛 Github 的時候,看到了一個項目 win12 —— 仿 Windows12 網頁版!被它實現的頁面功能震撼到了,大家可以一起來感受下! 首先是登錄頁面。 登錄後,再看看 “電腦桌面”,這風格完全和 Windows 操作系統的一模一樣!預設的應用程式有 ...
  • 一、定義 義一個對象來封裝一系列的對象交互。中介者模式使各對象不需要顯示地相互引用,從而使其耦合鬆散,而且可以讓你相對獨立地改變它們之間的交互。中介者模式又稱為調停模式,它是一種對象說行為型模式。 二、描述 在中介者模式中,引入了用於協調其他對象/類之間的相互調用的中介者類,為了讓系統具有更好的靈活 ...
  • C-08 聚合(分組)函數和SQL(DQL)執行流程 上一章,講到了SQL單行函數。實際上SQL函數還有一類,叫做聚合函數,或者多行,分組函數,它是對一組數據進行彙總的函數,輸入的是一組數據的集合,輸出的是單個值。 1.聚合函數的介紹 什麼是聚合函數 聚合函數作用於一組數據,並對一組數據返回一個值。 ...
  • 如題,這是一個很簡單也很基礎的問題,先看不同操作的結果有什麼不同 結果1:項目下出現了一個空“文件夾” 結果2:項目下多了一個“Python Package”,該package下包含了一個“__init__.py”文件,該py文件是空的 如果把該py文件刪掉後,package也就變成了directo ...
  • 1 數據可視化 自 DataGrip 2023.3 發佈以來,已整合 Lets-Plot 庫,實現數據可視化。該可視化功能可用於所有三種類型的網格: 主選項卡:在打開表、視圖或 CSV 文件時,在分割模式下顯示圖表。 結果選項卡:在 服務 工具視窗中觀察查詢結果時,可以顯示圖表而不是網格。 編輯器中 ...
  • 在Java中,我們經常需要獲取某個類的Class對象,以便進行反射操作、動態載入類或者獲取類的信息。本文將介紹Java中獲取Class對象的三種常見方式。 1. 使用.class語法 獲取Class對象最簡單的方式就是使用.class語法。每個Java類都有一個隱含的靜態成員變數class,可以通過 ...
  • C 語言中的類型轉換 有時,您必須將一種數據類型的值轉換為另一種類型。這稱為類型轉換 隱式轉換 當您將一種類型的值分配給另一種類型的變數時,編譯器會自動進行隱式轉換。 例如,如果您將一個 int 值分配給一個 float 類型: // 自動轉換:int to float float myFloat ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...