django源码分析 python manage.py runserver
django是一个快速开发web应用的框架, 笔者也在django框架上开发不少web应用,闲来无事,就想探究一下django底层到底是如何实现的,本文记录了笔者对django源码的分析过程
i believe to become a better developer you must get a better understanding of the underlying software systems you use on a daily basis and that includes programming languages, compilers and interpreters, databases and operating systems, web servers and web frameworks. and, to get a better and deeper understanding of those systems you must re-build them from scratch, brick by brick, wall by wall.
笔者摘抄了一段话,送给阅读本文的读者
正文
如何分析django源码,笔者选择从django项目的启动方式开始 python manage.py runserver,本文主要分析了django项目的启动流程
#!/usr/bin/env python import os import sys if __name__ == "__main__": os.environ.setdefault("django_settings_module", "order.settings") try: from django.core.management import execute_from_command_line except importerror: # the above import may fail for some other reason. ensure that the # issue is really that django is missing to avoid masking other # exceptions on python 2. try: import django except importerror: raise importerror( "couldn't import django. are you sure it's installed and " "available on your pythonpath environment variable? did you " "forget to activate a virtual environment?" ) raise execute_from_command_line(sys.argv)
在manage.py文件中,我们看到启动文件的入口是 excute_from_command_line(sys.argv)
def execute_from_command_line(argv=none): """ a simple method that runs a managementutility. """ utility = managementutility(argv) utility.execute()
这个函数是将命令行参数传递给了managementutility类,这个类的execute方法负责执行,这个方法主要是一些django的初始化参数的检查,以及通过sys.argv获取命令,得到相应的命令后,执行命令。
execute方法中的部分代码
... if settings.configured: # start the auto-reloading dev server even if the code is broken. # the hardcoded condition is a code smell but we can't rely on a # flag on the command class because we haven't located it yet. if subcommand == 'runserver' and '--noreload' not in self.argv: try: autoreload.check_errors(django.setup)() except exception: # the exception will be raised later in the child process # started by the autoreloader. pretend it didn't happen by # loading an empty list of applications. apps.all_models = defaultdict(ordereddict) apps.app_configs = ordereddict() apps.apps_ready = apps.models_ready = apps.ready = true ...
execute方法中有一段代码autoreload.check_errors(django.setup)(),会对django项目进行一些必要的初始化,并检查初始化的错误 django.setup()方法会注册项目app和配置日志文件,注册app即对settings.installed_apps中的app进行导入,并执行一些初始化方法
进行完所有初始化动作,运行命令
execute方法中的部分代码
... elif self.argv[1:] in (['--help'], ['-h']): sys.stdout.write(self.main_help_text() + '\n') else: self.fetch_command(subcommand).run_from_argv(self.argv) ...
self.fetch_command(subcommand)会返回一个basecommand类,主要是分析subcommand参数(subcommand是sys.argv里面获取到的),导入相应的命令类,最后返回类
我们通过分析,runserver参数最终获取到的命令类是django/contrib/staticfiles/management/command/runserver.py 里的command类
这是command类的继承关系图。command类通过run_from_argv(self.argv)执行命令
basecommand类中run_from_argv方法的部分代码
... try: self.execute(*args, **cmd_options) except exception as e: if options.traceback or not isinstance(e, commanderror): raise ...
run_from_argv(self.argv)方法中主要通过execute()来继续执行,excute中会对django项目进行检查,然后通过self.handle()继续执行
runservercommand类里面的handle方法部分代码
def handle(self, *args, **options):
...
if not self.addr: self.addr = '::1' if self.use_ipv6 else '127.0.0.1' self._raw_ipv6 = self.use_ipv6 self.run(**options)
handle()方法里面也进行了一些检查,然后继续执行self.run()来启动服务器
runservercommand中的部分代码 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) def inner_run(self, *args, **options): ... try: handler = self.get_handler(*args, **options) run(self.addr, int(self.port), handler, ipv6=self.use_ipv6, threading=threading, server_cls=self.server_cls) except socket.error as e: ...
run方法中选择了启动的解释器,最后都是通过inner_run中的run方法来执行,会启动一个wsgiserver, wsgiserver需要一个回调函数handler(或者application),来执行django视图里面代码。
至此,django项目服务器启动流程完毕,接下来就是这个简单的wsgiserver,接受请求,解析请求参数,将请求参数传递给回调函数handler(或者application),handler根据参数执行相应的代码,返回数据给wsgiserver,wsgiserver最终将数据返回给浏览器。
关于wsgi可以参考这篇文章,理解python wsgi
推荐阅读
-
python中pygame针对游戏窗口的显示方法实例分析(附源码)
-
Python的socket模块源码中的一些实现要点分析
-
Python的socket模块源码中的一些实现要点分析
-
python中pygame针对游戏窗口的显示方法实例分析(附源码)
-
分析Python的Django框架的运行方式及处理流程
-
Python Django框架实现商城项目源码加设计文档和注释
-
Django-restframework 源码分析之认证
-
Python及Django框架生成二维码的方法分析
-
Python日志打印里logging.getLogger源码分析详解
-
详解Django-restframework 之频率源码分析