Python實(shí)現(xiàn)模塊熱加載的示例代碼
為什么需要熱加載
在某些情況,你可能不希望關(guān)閉Python進(jìn)程并重新打開,或者你無法重新啟動Python,這時候就需要實(shí)現(xiàn)實(shí)時修改代碼實(shí)時生效,而不用重新啟動Python
在我的需求下,這個功能非常重要,我將Python注入到了其他進(jìn)程,并作為一個線程運(yùn)行。如果我想關(guān)閉Python,要么殺死Python相關(guān)的線程,要么重新啟動進(jìn)程,這都比較麻煩。所以當(dāng)我修改完代碼后,熱加載代碼是最方便的方法
Python中的導(dǎo)入機(jī)制
我們重復(fù)導(dǎo)入一個庫時,第二次導(dǎo)入時并沒有運(yùn)行庫里面的代碼,比如先寫一個a.py
,在里面寫一行代碼print("a模塊加載")
,然后在寫一個b.py
, 里面寫兩行import a
。即使你在多線程中再導(dǎo)入一遍a模塊,也不會打印。例如下面的代碼:
import a import threading print(id(a)) def test(): import a print(id(a)) threading.Thread(target=test).start()
可以看到a的id是一樣的,也就是同一個對象。
為什么會這樣呢?這和Python的模塊導(dǎo)入機(jī)制有關(guān),Python會在sys.modules
這個字典里存儲著所有的全局模塊,當(dāng)你導(dǎo)入一個新模塊時,他會先查找sys.modules
里有沒有這個模塊,如果沒有再導(dǎo)入,如果有就在當(dāng)前代碼增加個引用。舉個最簡單的例子:
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
先導(dǎo)入a模塊,這樣sys.modules
已經(jīng)有了a模塊,你就可以使用sys.modules["a"]
來使用a模塊,它和import a
基本是一樣的。如果你先import b
就會發(fā)現(xiàn)sys.modules
不存在a
重新導(dǎo)入模塊1
既然知道它是先查找sys.modules
,那我在導(dǎo)入之前,先刪除掉里面的a再導(dǎo)入就可以了
import a import sys del sys.modules["a"] import a
這樣就能重新加載模塊
重新導(dǎo)入模塊2
Python基礎(chǔ)庫也提供了一個方法重新加載模塊:
import a import importlib importlib.reload(a)
看一下內(nèi)部代碼是怎么實(shí)現(xiàn)的:
邏輯也比較簡單, 先看sys.modules
里有沒有這個模塊,如果有就使用_bootstrap._exec
導(dǎo)入模塊。我們是不是也可以通過_bootstrap._exec
來重新導(dǎo)入模塊,可以但不建議,因?yàn)橄聞澗€開頭的模塊或者函數(shù)都是不建議外部使用的,這些接口可能在版本更新后變動比較頻繁
無法熱加載的情況
__main__
模塊無法熱加載。當(dāng)你執(zhí)行python a.py
,這個a.py文件是無法熱加載的,它并沒有作為模塊導(dǎo)入,在sys.modules
的名稱就是__main__
如果你在__main__
使用from a import A
導(dǎo)入的類,即使a模塊重新加載,__main__
里面的A也不會改變
熱加載無法影響已經(jīng)實(shí)例化的對象,比如你修改了模塊里面的類代碼,但是已經(jīng)在__main__
里實(shí)例化了這個類對象,并且一直使用未釋放,它的邏輯在熱加載之后不會受影響。
函數(shù)級熱加載
要想實(shí)現(xiàn)函數(shù)、方法乃至對象級別的熱加載,得修改內(nèi)存中的Python對象。有一個項(xiàng)目實(shí)現(xiàn)了這種,有興趣的可以看:https://github.com/breuleux/jurigged
我的需求沒有這么細(xì),就不測試了
監(jiān)聽文件變化
我選擇的是watchdog
,另一個pyinotify
不支持Windows。
watchdog
在Windows上有點(diǎn)小bug,修改文件會觸發(fā)兩次事件。搜到一個解決方案:不使用默認(rèn)的事件觸發(fā),而是利用文件快照,每隔一段時間做一次比對。原文鏈接:Python神器watchdog(監(jiān)控文件變化),我測試了一下效果很好。
源碼
完整的源碼就不放了,具體可以看:https://github.com/kanadeblisst00/module_hot_loading
國內(nèi)倉庫: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的參數(shù):
- 需要監(jiān)控的目錄路徑
- 停止監(jiān)控的事件信號
- __main__的代碼文件路徑
- interval: 每隔幾秒打一次文件快照做比對
- only_import_exist: 只重新加載已經(jīng)導(dǎo)入的模塊
效果
到此這篇關(guān)于Python實(shí)現(xiàn)模塊熱加載的示例代碼的文章就介紹到這了,更多相關(guān)Python模塊熱加載內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python遞歸調(diào)用中的坑:打印有值, 返回卻None
這篇文章主要介紹了python遞歸調(diào)用中的坑:打印有值, 返回卻None,本文通過問題分析給出解決方法,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-03-03pycharm實(shí)現(xiàn)print輸出保存到txt文件
這篇文章主要介紹了pycharm實(shí)現(xiàn)print輸出保存到txt文件,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-06-06python實(shí)現(xiàn)簡單的TCP代理服務(wù)器
這篇文章主要介紹了python實(shí)現(xiàn)簡單的TCP代理服務(wù)器,包含了完整的實(shí)現(xiàn)過程及對應(yīng)的源碼與說明文檔下載,非常具有參考借鑒價值,需要的朋友可以參考下2014-10-10Python基于生成器迭代實(shí)現(xiàn)的八皇后問題示例
這篇文章主要介紹了Python基于生成器迭代實(shí)現(xiàn)的八皇后問題,簡單描述了八皇后問題,并結(jié)合實(shí)例形式分析了Python基于生成器迭代解決八皇后問題的相關(guān)操作技巧,需要的朋友可以參考下2018-05-05