Python重新加載模塊的實(shí)現(xiàn)方法
importlib 模塊的作用
模塊,是一個(gè)一個(gè)單獨(dú)的py文件 包,里面包含多個(gè)模塊(py文件)
動(dòng)態(tài)導(dǎo)入模塊,這樣就不用寫那么多的import代碼, 典型的例子: 自動(dòng)同步服務(wù),每個(gè)網(wǎng)站都有一個(gè)py文件。主進(jìn)程里收到同步任務(wù),根據(jù)名稱來動(dòng)態(tài)導(dǎo)入對(duì)應(yīng)的py文件,這樣就不用寫那么多的import代碼。(有點(diǎn)類似java的工廠方法)
但是,importlib并不能解決我在線修改py源碼,再不重啟進(jìn)程的情況下,使修改生效。 這種情況,可以使用reload()
reload方法
為防止兩個(gè)模塊互相導(dǎo)入的問題,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是無法影響另一個(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)聽redis消息,收到消息,即執(zhí)行reload方法
p = Process(target=reload_spider)
p.start()
plist.append(p)
for p in plist:
p.join()
# 監(jiān)聽redis,執(zhí)行reload方法
def reload_spider():
""" 監(jiān)聽文件變化,自動(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)程中,多線程,任一線程進(jìn)行了reload操作,其他線程均受影響
def pre_run():
t = threading.Thread(target=reload_spider, name='LoopThread')
t.start()
# t.join()
""" 在每個(gè)進(jì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)前線程的名字
current_process_name = multiprocessing.current_process().name
logger.debug('當(dāng)前進(jìn)程名稱:%s - pid: %s' % (current_process_name, os.getpid()))
logger.debug('進(jìn)程 %s,收到消息: %s' % (current_process_name, body))
# 收到任務(wù)消息,丟給線程池處理
pool.apply_async(run, (properties, body, mq_service))
# 開始監(jiān)聽入口通道
mq_service.receive(callback)
reload_spider中監(jiān)聽redi中的消息,如果有reload標(biāo)識(shí),進(jìn)行reload操作
def reload_spider():
""" 監(jiān)聽文件變化,自動(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)測試,其他線程中的引入的變量,也改變了。
當(dāng)然,消息監(jiān)聽最好使用mq或者是redis阻塞隊(duì)列
以上這篇Python重新加載模塊的實(shí)現(xiàn)方法就是小編分享給大家的全部內(nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
跟老齊學(xué)Python之玩轉(zhuǎn)字符串(3)
字符串是一個(gè)很長的話題,縱然現(xiàn)在開始第三部分,但是也不能完全說盡。因?yàn)樽址亲匀徽Z言中最復(fù)雜的東西,也是承載功能最多的,計(jì)算機(jī)高級(jí)語言編程,要解決自然語言中的問題,讓自然語言中完成的事情在計(jì)算機(jī)上完成,所以,也不得不有更多的話題。2014-09-09
基于opencv和pillow實(shí)現(xiàn)人臉識(shí)別系統(tǒng)(附demo)
人臉識(shí)別就是一個(gè)程序能識(shí)別給定圖像或視頻中的人臉,本文主要介紹了opencv和pillow實(shí)現(xiàn)人臉識(shí)別系統(tǒng),本文不涉及分類器、訓(xùn)練識(shí)別器等算法原理,感興趣的可以了解一下2021-11-11
Pandas數(shù)據(jù)合并的兩種實(shí)現(xiàn)方法
本文主要介紹了Pandas數(shù)據(jù)合并的兩種實(shí)現(xiàn)方法,DataFrame數(shù)據(jù)合并主要使用merge()方法和concat()方法,具有一定的參考價(jià)值,感興趣的可以了解一下2023-11-11
Python2和Python3.6環(huán)境解決共存問題
這篇文章主要介紹了Python2和Python3.6環(huán)境解決共存問題,需要的朋友可以參考下2018-11-11
使用Python實(shí)現(xiàn)簡單的數(shù)據(jù)備份
數(shù)據(jù)備份,即數(shù)據(jù)的復(fù)制和存儲(chǔ),是指將數(shù)據(jù)從一個(gè)位置復(fù)制到另一個(gè)位置,以防止原始數(shù)據(jù)丟失或損壞,下面我們就來了解一下用Python如何實(shí)現(xiàn)這一功能吧2025-03-03
在ipython notebook中使用argparse方式
這篇文章主要介紹了在ipython notebook中使用argparse方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-04-04

