在開發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
新聞熱點
疑難解答