Python重新加載模塊的實(shí)現(xiàn)方法
importlib 模塊的作用
模塊,是一個(gè)一個(gè)單獨(dú)的py文件 包,里面包含多個(gè)模塊(py文件)
動(dòng)態(tài)導(dǎo)入模塊,這樣就不用寫(xiě)那么多的import代碼, 典型的例子: 自動(dòng)同步服務(wù),每個(gè)網(wǎng)站都有一個(gè)py文件。主進(jìn)程里收到同步任務(wù),根據(jù)名稱(chēng)來(lái)動(dòng)態(tài)導(dǎo)入對(duì)應(yīng)的py文件,這樣就不用寫(xiě)那么多的import代碼。(有點(diǎn)類(lèi)似java的工廠方法)
但是,importlib并不能解決我在線(xiàn)修改py源碼,再不重啟進(jìn)程的情況下,使修改生效。 這種情況,可以使用reload()
reload方法
為防止兩個(gè)模塊互相導(dǎo)入的問(wèn)題,Python默認(rèn)所有的模塊都只導(dǎo)入一次,如果需要重新導(dǎo)入模塊, Python2.7可以直接用reload(),Python3可以用下面幾種方法:
方法一:基本方法 from imp import reload reload(module)
方法二:按照套路,可以這樣 import imp imp.reload(module)
方法三:看看imp.py,有發(fā)現(xiàn),所以還可以這樣 import importlib importlib.reload(module)
方法四:根據(jù)天理,當(dāng)然也可以這樣 from importlib import reload reload(module)
在多進(jìn)程的 程序中,一個(gè)進(jìn)程的reload是無(wú)法影響另一個(gè)進(jìn)程的
例子:
# 在主進(jìn)程中啟動(dòng)多進(jìn)程 def begin(): """ 啟動(dòng)多進(jìn)程 """ plist = [] for i in xrange(Num_process): p = Process(target=pre_run) p.start() plist.append(p) # 此進(jìn)程監(jiān)聽(tīng)redis消息,收到消息,即執(zhí)行reload方法 p = Process(target=reload_spider) p.start() plist.append(p) for p in plist: p.join()
# 監(jiān)聽(tīng)redis,執(zhí)行reload方法 def reload_spider(): """ 監(jiān)聽(tīng)文件變化,自動(dòng)reload """ rconn = redis.Redis(host=settings.REDIS_HOST, port=settings.REDIS_PORT, db=settings.REDIS_DB, password=settings.REDIS_PW) while True: try: key = 'reload-spider' value = rconn.get(key) print value if value == '1': crawler_module = importlib.import_module('crawlers.%s' % 'temp'.lower()) reload(crawler_module) crawlerClass = getattr(crawler_module, 'temp'.upper()) print 'reload_spider 中的class: %s' % (crawlerClass.name) # rconn.delete(key) except Exception, e: pass time.sleep(3)
另一個(gè)進(jìn)程打印py文件里面一個(gè)變量
crawler = get_crawler_from_factory(mq_service, message) print crawler.name
結(jié)果發(fā)現(xiàn),一個(gè)進(jìn)程中進(jìn)行了reload,并不能改變另外一個(gè)進(jìn)程中的變量。那么在同一個(gè)進(jìn)程中呢...
同一進(jìn)程中,多線(xiàn)程,任一線(xiàn)程進(jìn)行了reload操作,其他線(xiàn)程均受影響
def pre_run(): t = threading.Thread(target=reload_spider, name='LoopThread') t.start() # t.join() """ 在每個(gè)進(jìn)程里面再使用多線(xiàn)程 """ pool = ThreadPool(Num_Thread) # 初始化mq通道 mq_service = RabbitMqService() def callback(ch, method, properties, body): # 消息確認(rèn) mq_service.input_channel.basic_ack(delivery_tag=method.delivery_tag) # 獲取當(dāng)前線(xiàn)程的名字 current_process_name = multiprocessing.current_process().name logger.debug('當(dāng)前進(jìn)程名稱(chēng):%s - pid: %s' % (current_process_name, os.getpid())) logger.debug('進(jìn)程 %s,收到消息: %s' % (current_process_name, body)) # 收到任務(wù)消息,丟給線(xiàn)程池處理 pool.apply_async(run, (properties, body, mq_service)) # 開(kāi)始監(jiān)聽(tīng)入口通道 mq_service.receive(callback)
reload_spider中監(jiān)聽(tīng)redi中的消息,如果有reload標(biāo)識(shí),進(jìn)行reload操作
def reload_spider(): """ 監(jiān)聽(tīng)文件變化,自動(dòng)reload """ rconn = redis.Redis(host=settings.REDIS_HOST, port=settings.REDIS_PORT, db=settings.REDIS_DB, password=settings.REDIS_PW) while True: try: key = 'reload-spider' value = rconn.get(key) print value if value == '1': crawler_module = importlib.import_module('crawlers.%s' % 'temp'.lower()) reload(crawler_module) crawlerClass = getattr(crawler_module, 'temp'.upper()) print 'reload_spider 中的class: %s' % (crawlerClass.name) # rconn.delete(key) except Exception, e: pass time.sleep(3)
經(jīng)測(cè)試,其他線(xiàn)程中的引入的變量,也改變了。
當(dāng)然,消息監(jiān)聽(tīng)最好使用mq或者是redis阻塞隊(duì)列
以上這篇Python重新加載模塊的實(shí)現(xiàn)方法就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- Python動(dòng)態(tài)加載模塊的3種方法
- Python實(shí)現(xiàn)動(dòng)態(tài)加載模塊、類(lèi)、函數(shù)的方法分析
- 跟老齊學(xué)Python之模塊的加載
- Python模塊的加載講解
- Python自動(dòng)重新加載模塊詳解(autoreload module)
- Python之inspect模塊實(shí)現(xiàn)獲取加載模塊路徑的方法
- Python如何重新加載模塊
- Python新手如何理解循環(huán)加載模塊
- Python 關(guān)于模塊和加載模塊的實(shí)現(xiàn)
- 簡(jiǎn)單談?wù)凱ython中的模塊導(dǎo)入
相關(guān)文章
Python線(xiàn)程之定位與銷(xiāo)毀的實(shí)現(xiàn)
這篇文章主要介紹了Python線(xiàn)程之定位與銷(xiāo)毀的實(shí)現(xiàn),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2019-02-02對(duì)Xpath 獲取子標(biāo)簽下所有文本的方法詳解
今天小編就為大家分享一篇對(duì)Xpath 獲取子標(biāo)簽下所有文本的方法詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-01-01Matplotlib學(xué)習(xí)筆記之plt.xticks()用法
在matplotlib中ticks表示的是刻度,而刻度有兩層意思,一個(gè)是刻標(biāo)(locs),一個(gè)是刻度標(biāo)簽(tick?labels),下面這篇文章主要給大家介紹了關(guān)于Matplotlib學(xué)習(xí)筆記之plt.xticks()用法的相關(guān)資料,需要的朋友可以參考下2022-09-09Python爬蟲(chóng)基礎(chǔ)之初次使用scrapy爬蟲(chóng)實(shí)例
今天給大家?guī)?lái)的是關(guān)于Python爬蟲(chóng)的相關(guān)知識(shí),文章圍繞著Python scrapy展開(kāi),文中有非常詳細(xì)的介紹及代碼示例,需要的朋友可以參考下2021-06-06Django框架靜態(tài)文件使用/中間件/禁用ip功能實(shí)例詳解
這篇文章主要介紹了Django框架靜態(tài)文件使用/中間件/禁用ip功能,結(jié)合實(shí)例形式詳細(xì)分析了Django框架靜態(tài)文件的使用、中間件的原理、操作方法以及禁用ip功能相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2019-07-07web.py在SAE中的Session問(wèn)題解決方法(使用mysql存儲(chǔ))
這篇文章主要介紹了web.py在SAE中的Session問(wèn)題解決方法(使用mysql存儲(chǔ)),本文直接給出實(shí)現(xiàn)代碼,代碼中包含詳細(xì)注釋,需要的朋友可以參考下2015-06-06python實(shí)現(xiàn)sm2和sm4國(guó)密(國(guó)家商用密碼)算法的示例
這篇文章主要介紹了python實(shí)現(xiàn)sm2和sm4國(guó)密(國(guó)家商用密碼)算法的示例,幫助大家使用python加密文件,感興趣的朋友可以了解下2020-09-09Python使用pandas導(dǎo)入csv文件內(nèi)容的示例代碼
這篇文章主要介紹了Python使用pandas導(dǎo)入csv文件內(nèi)容,本文結(jié)合示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-12-12Python3中的最大整數(shù)和最大浮點(diǎn)數(shù)實(shí)例
今天小編就為大家分享一篇Python3中的最大整數(shù)和最大浮點(diǎn)數(shù)實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-07-07