Python重新加载模块方法

importlib 模块的作用

模块,是一个一个单独的py文件 包,里面包含多个模块(py文件)

动态导入模块,这样就不用写那么多的import代码, 典型的例子: 自动同步服务,每个网站都有一个py文件。主进程里收到同步任务,根据名称来动态导入对应的py文件,这样就不用写那么多的import代码。(有点类似java的工厂方法)

但是,importlib并不能解决我在线修改py源码,再不重启进程的情况下,使修改生效。 这种情况,可以使用reload()

reload方法

为防止两个模块互相导入的问题,Python默认所有的模块都只导入一次,如果需要重新导入模块, Python2.7可以直接用reload(),Python3可以用下面几种方法:

方法一:基本方法 from imp import reload reload(module)

方法二:按照套路,可以这样 import imp imp.reload(module)

方法三:看看imp.py,有发现,所以还可以这样 import importlib importlib.reload(module)

方法四:根据天理,当然也可以这样 from importlib import reload reload(module)

在多进程的 程序中,一个进程的reload是无法影响另一个进程的

例子:

# 在主进程中启动多进程
def begin():
    """ 启动多进程 """
    plist = []
    for i in xrange(Num_process):
        p = Process(target=pre_run)
        p.start()
        plist.append(p)
    # 此进程监听redis消息,收到消息,即执行reload方法
    p = Process(target=reload_spider)
    p.start()
    plist.append(p)
    for p in plist:
        p.join()
# 监听redis,执行reload方法
def reload_spider():
    """ 监听文件变化,自动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)

另一个进程打印py文件里面一个变量

crawler = get_crawler_from_factory(mq_service, message)
    print crawler.name

结果发现,一个进程中进行了reload,并不能改变另外一个进程中的变量。那么在同一个进程中呢...

同一进程中,多线程,任一线程进行了reload操作,其他线程均受影响

def pre_run():
    t = threading.Thread(target=reload_spider, name='LoopThread')
    t.start()
    # t.join()

    """ 在每个进程里面再使用多线程 """
    pool = ThreadPool(Num_Thread)
    # 初始化mq通道
    mq_service = RabbitMqService()

    def callback(ch, method, properties, body):
        # 消息确认
        mq_service.input_channel.basic_ack(delivery_tag=method.delivery_tag)
        # 获取当前线程的名字
        current_process_name = multiprocessing.current_process().name
        logger.debug('当前进程名称:%s - pid: %s' % (current_process_name, os.getpid()))
        logger.debug('进程 %s,收到消息: %s' % (current_process_name, body))
        # 收到任务消息,丢给线程池处理
        pool.apply_async(run, (properties, body, mq_service))
    # 开始监听入口通道
    mq_service.receive(callback)
    

reload_spider中监听redi中的消息,如果有reload标识,进行reload操作

def reload_spider():
    """ 监听文件变化,自动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)

经测试,其他线程中的引入的变量,也改变了。

当然,消息监听最好使用mq或者是redis阻塞队列

Python中,你可以使用`import`关键字来加载Turtle模块加载模块后,你就可以使用其中定义的函数、类和变量了。 下面是一些加载Turtle模块的示例代码: ```python import turtle # 现在你可以使用turtle模块中的函数和类了 my_turtle = turtle.Turtle() my_turtle.forward(100) turtle.done() ``` 在这个例子中,我们使用`import turtle`语句将Turtle模块加载Python程序中。然后,我们创建了一个Turtle对象,并调用了该对象的`forward`方法,使其向前移动100个单位。最后,我们使用`turtle.done()`来保持图形窗口的显示。 你也可以使用`import`语句给模块起一个别名,以便在代码中更方便地使用。例如: ```python import turtle as t my_turtle = t.Turtle() my_turtle.forward(100) t.done() ``` 在这个例子中,我们将Turtle模块的别名设置为`t`,然后使用`t.Turtle()`来创建Turtle对象。 除了使用`import`语句加载整个模块,你还可以使用`from`语句从模块中导入特定的函数、类或变量。例如: ```python from turtle import Turtle my_turtle = Turtle() my_turtle.forward(100) ``` 在这个例子中,我们只从Turtle模块中导入了Turtle类,而不是整个模块。这样,在代码中就可以直接使用`Turtle()`来创建Turtle对象,而不需要加上模块名。 希望这些示例能帮助你加载并使用Turtle模块。如果你需要更多关于Turtle模块的详细信息,你可以查阅Python官方文档或者Turtle模块的文档。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值