首页 > 编程 > Python > 正文

浅谈django开发者模式中的autoreload是如何实现的

2020-02-16 02:06:45
字体:
来源:转载
供稿:网友

在开发django应用的过程中,使用开发者模式启动服务是特别方便的一件事,只需要 python manage.py runserver 就可以运行服务,并且提供了非常人性化的autoreload机制,不需要手动重启程序就可以修改代码并看到反馈。刚接触的时候觉得这个功能比较人性化,也没觉得是什么特别高大上的技术。后来有空就想着如果是我来实现这个autoreload会怎么做,想了很久没想明白,总有些地方理不清楚,看来第一反应真是眼高手低了。于是就专门花了一些时间研究了django是怎样实现autoreload的,每一步都看源码说话,不允许有丝毫的想当然:

1、runserver命令。在进入正题之前其实有一大段废话,是关于runserver命令如何执行的,和主题关系不大,就简单带一下:

命令行键入 python manage.py runserver 后,django会去寻找runserver这个命令的执行模块,最后落在

django/contrib/staticfiles/management/commands/runserver.py模块上:

#django/contrib/staticfiles/management/commands/runserver.pyfrom django.core.management.commands.runserver import /Command as RunserverCommandclass Command(RunserverCommand):help = "Starts a lightweight Web server for development and also serves static files."

而这个Command的执行函数在这:

#django/core/management/commands/runserver.pyclass Command(BaseCommand):  def run(self, **options):  """  Runs the server, using the autoreloader if needed  """  use_reloader = options['use_reloader']  if use_reloader:    autoreload.main(self.inner_run, None, options)  else:    self.inner_run(None, **options)

这里有关于use_reloader的判断。如果我们在启动命令中没有加--noreload,程序就会走autoreload.main这个函数,如果加了,就会走self.inner_run,直接启动应用。

其实从autoreload.main的参数也可以看出,它应该是对self.inner_run做了一些封装,autoreload的机制就在这些封装当中,下面我们继续跟。

PS: 看源码的时候发现django的command模式还是实现的很漂亮的,值得学习。

2、autoreload模块。看autoreload.main():

#django/utils/autoreload.py:def main(main_func, args=None, kwargs=None):  if args is None:    args = ()  if kwargs is None:    kwargs = {}  if sys.platform.startswith('java'):    reloader = jython_reloader  else:    reloader = python_reloader  wrapped_main_func = check_errors(main_func)  reloader(wrapped_main_func, args, kwargs)

这里针对jpython和其他python做了区别处理,先忽略jpython;check_errors就是把对main_func进行错误处理,也先忽略。看python_reloader:

#django/utils/autoreload.py:def python_reloader(main_func, args, kwargs):  if os.environ.get("RUN_MAIN") == "true":    thread.start_new_thread(main_func, args, kwargs)    try:      reloader_thread()    except KeyboardInterrupt:      pass  else:    try:      exit_code = restart_with_reloader()      if exit_code < 0:        os.kill(os.getpid(), -exit_code)      else:        sys.exit(exit_code)    except KeyboardInterrupt:      pass            
发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表