文档章节

django源码分析--03app加载过程

AllenOR灵感
 AllenOR灵感
发布于 2017/09/10 01:18
字数 917
阅读 1
收藏 0

django.core.management.__init__.ManagementUtility.execute方法中通过autoreload.check_errors(django.setup)()这一行代码来加载django的app。

django.__init__.setup方法中通过django.apps.registry.Apps.populate(settings.INSTALLED_APPS)来加载所有配置文件中定义好的app。刚创建好的一个project默认模版都会在settings.py文件中指定好这几个app:

django.contrib.admin
django.contrib.auth
django.contrib.contenttypes
django.contrib.sessions
django.contrib.messages
django.contrib.staticfiles

django.apps.registry.Apps.populate方法中通过下面这个代码片段调用django.apps.config.AppConfig.create的类方法来加载app和实例化该app的类对象。

django.apps.registry.py
from .config import AppConfig


class Apps(object):

    def __init__(self, installed_apps=()):
        if installed_apps is None and hasattr(sys.modules[__name__], 'apps'):
            raise RuntimeError("You must supply an installed_apps argument.")

        self.all_models = defaultdict(OrderedDict)
        self.app_configs = OrderedDict()
        self.stored_app_configs = []
        self.apps_ready = self.models_ready = self.ready = False
        self._lock = threading.Lock()
        self._pending_operations = defaultdict(list)
        if installed_apps is not None:
            self.populate(installed_apps)

    def populate(self, installed_apps=None):
        if self.ready:
            return

        with self._lock:
            if self.ready:
                return

            if self.app_configs:
                raise RuntimeError("populate() isn't reentrant")

            for entry in installed_apps:
                if isinstance(entry, AppConfig):
                    app_config = entry
                else:
                    # 这里这个for循环落实的工作是,
                    # 实例化每个app的配置管理对象(
                    # django.contrib.auth.apps.AuthConfig()),
                    # 实例化过后默认有很多参数都是None,
                    # 例如models_module、models等。
                    app_config = AppConfig.create(entry)
                if app_config.label in self.app_configs:
                    raise ImproperlyConfigured(
                        "Application labels aren't unique, "
                        "duplicates: %s" % app_config.label)

                self.app_configs[app_config.label] = app_config

            counts = Counter(
                app_config.name for app_config in self.app_configs.values())
            duplicates = [
                name for name, count in counts.most_common() if count > 1]
            if duplicates:
                raise ImproperlyConfigured(
                    "Application names aren't unique, "
                    "duplicates: %s" % ", ".join(duplicates))

            self.apps_ready = True

            # Load models.
            for app_config in self.app_configs.values():
                # 这里采用字典引用的方式赋值给all_models,
                # 也就是说后续self.all_models['django.contrib.auth']
                # 发生变化,它也会发生变化。            
                all_models = self.all_models[app_config.label] 

                # 这里采用了非常高级的做法,以我现在的水平,
                # 暂时还不能完全理解,但是我已经有一些眉目了,
                # 详细的说明请进入import_models方法中继续查看。         
                app_config.import_models(all_models)                

            self.clear_cache()

            self.models_ready = True

            for app_config in self.get_app_configs():
                # 这里是针对app进行检查是否已经存在,
                # 如果已经存在则不做事情,如果不存在则
                # 将它添加到CheckRegistry.registered_checks和
                # 将它添加到CheckRegistry.deployment_checks的
                # 注册列表中。
                # 备注: 维护这个列表的用意我暂时还不清楚。
                app_config.ready()

            self.ready = True
django.apps.config.py
class AppConfig(object):

    def __init__(self, app_name, app_module):
        self.name = app_name
        self.module = app_module

    @classmethod
    def create(cls, entry):
        try:
            # entry = 'django.contrib.auth'   ,   module = <module 'django.contrib.auth' from 'C:\\Python35\\lib\\site-packages\\django\\contrib\\auth\\__init__.py'>
            module = import_module(entry)               
        except ImportError:
            module = None
            mod_path, _, cls_name = entry.rpartition('.')
            if not mod_path:
                raise
        else:
            try:
                # entry = 'django.contrib.auth.apps.AuthConfig'
                entry = module.default_app_config                       
            except AttributeError:
                return cls(entry, module)
            else:
                # mod_path = 'django.contrib.auth.apps', 
                # _ = '.' ,
                # cls_name = 'AuthConfig'
                mod_path, _, cls_name = entry.rpartition('.')           

        # mod = <module 'django.contrib.auth.apps' from 'C:\\Python35\\lib\\site-packages\\django\\contrib\\auth\\apps.py'>
        mod = import_module(mod_path)                                   
        try:
            # cls = <class django.contrib.auth.apps.AuthConfig>
            cls = getattr(mod, cls_name)                                
        except AttributeError:
            if module is None:
                import_module(entry)
            else:
                raise

        if not issubclass(cls, AppConfig):
            raise ImproperlyConfigured(
                "'%s' isn't a subclass of AppConfig." % entry)

        try:
            # app_name = cls.name = 'django.contrib.auth'
            app_name = cls.name                                         
        except AttributeError:
            raise ImproperlyConfigured(
                "'%s' must supply a name attribute." % entry)

        # app_module = <module 'django.contrib.auth' from 'C:\\Python35\\lib\\site-packages\\django\\contrib\\auth\\__init__.py'>
        app_module = import_module(app_name)                            

        # <AuthConfig: auth> = <django.contrib.auth.apps.AuthConfig object app_name='django.contrib.auth' app_module='<module 'django.contrib.auth' from 'C:\\Python35\\lib\\site-packages\\django\\contrib\\auth\\__init__.py'>' at 0x000001E02D9AE470>
        return cls(app_name, app_module)                              


    def import_models(self, all_models):
        self.models = all_models

        if module_has_submodule(self.module, MODELS_MODULE_NAME):
            # self.name = app_name = 'django.contrib.auth'  , MODELS_MODULE_NAME = 'models' ; models_module_name = 'django.contrib.auth.models'
            models_module_name = '%s.%s' % (self.name, MODELS_MODULE_NAME)          

            # 这里将'django.contrib.auth.models'整个文件模块形式的导入到self.models_module变量中.
            # 但是由于'django.contrib.auth.models'的class中有继承ModelBase,然后ModelBase.__new__,
            # 在导入过程中会被触发并执行__new__下的代码块,该代码块中对
            # apps.get_containing_app_config('django.contrib.auth')单独进行register_model(模型注册),
            # 因此它会再次更新self.all_models,然后all_models会跟着发生变化,然后这里的self.models也会跟着发生变化。
            # 最终呈现出来的是: <django.contrib.auth.AppConfig object self.models=OrderedDict([('permission', <class 'django.contrib.auth.models.Permission'>), ('group_permissions', <class 'django.contrib.auth.models.Group_permissions'>), ('group', <class 'django.contrib.auth.models.Group'>), ('user_groups', <class 'django.contrib.auth.models.User_groups'>), ('user_user_permissions', <class 'django.contrib.auth.models.User_user_permissions'>), ('user', <class 'django.contrib.auth.models.User'>)]) at 0x000001E02D9AE470>
            # 通过呈现结果的对象来看,self.models已经装载了这些<已经加载好的数据库>对象。
            self.models_module = import_module(models_module_name)     
            # self.models_module = import_module(models_module_name) = import_module('django.contrib.auth.models')

总结
django在启动wsgi之前,会调用django.apps模块来读取项目文件中的settings.py拿到这面这几个app,然后交给django.apps的registry.py和config.py来进行统一的配置加载、实例化(含models数据库对象)。

本文转载自:http://www.jianshu.com/p/952ab113d638

共有 人打赏支持
AllenOR灵感
粉丝 10
博文 2635
码字总数 83001
作品 0
程序员
Django1.8 django-admin和manage命令简要分析

startproject和startapp 创建一个项目(Project) Django使用django-admin startproject [projectname]命令生成project。 django-admin命令行的路径为:django/bin/django-admin.py 以djang......

donhui
2015/11/23
1K
3
django项目在uwsgi+nginx上部署遇到的坑

本文来自网易云社区 作者:王超 问题背景 django框架提供了一个开发调试使用的WSGIServer, 使用这个服务器可以很方便的开发web应用。但是 正式环境下却不建议使用这个服务器, 其性能、安全性...

网易云
09/30
0
0
Spring MVC 原理探秘 - 容器的创建过程

1.简介 在上一篇文章中,我向大家介绍了 Spring MVC 是如何处理 HTTP 请求的。Spring MVC 可对外提供服务时,说明其已经处于了就绪状态。再次之前,Spring MVC 需要进行一系列的初始化操作。...

coolblog.xyz
07/03
0
0
newrelic python agent 源码分析-1

Newrelic 是APM(Application Performance Management)(应用性能管理/监控)解决方案提供商。项目中,通常用它来追踪应用的性能。最近看了一下 newrelic-python-agent 源码,这是查看源码过...

goodspeed
05/16
0
0
Django 结构及处理流程分析

conf 主要有两个作用:1) 处理全局配置, 比如数据库、加载的应用、 MiddleWare等 2) 处理urls配置, 就是url与view的映射关系。 contrib (贡献) 由Django的开发者贡献的功能模块,不过既然都已...

岭南六少
2011/08/08
0
0

没有更多内容

加载失败,请刷新页面

加载更多

linux 系统的运行级别

运行级别 运行级别 | 含义 0 关机 1 单用户模式,可以想象为windows 的安全模式,主要用于修复系统 2 不完全的命令模式,不含NFS服务 3 完全的命令行模式,就是标准的字符界面 4 系统保留 5 ...

Linux学习笔记
27分钟前
0
0
学习设计模式——命令模式

任何模式的出现,都是为了解决一些特定的场景的耦合问题,以达到对修改封闭,对扩展开放的效果。命令模式也不例外: 命令模式是为了解决命令的请求者和命令的实现者之间的耦合关系。 解决了这...

江左煤郎
34分钟前
2
0
字典树收集(非线程安全,后续做线程安全改进)

将500W个单词放进一个数据结构进行存储,然后进行快速比对,判断一个单词是不是这个500W单词之中的;来了一个单词前缀,给出500w个单词中有多少个单词是该前缀. 1、这个需求首先需要设计好数据结...

算法之名
昨天
8
0
GRASP设计模式

此文参考了这篇博客,建议读者阅读原文。 面向对象(Object-Oriented,OO)是当下软件开发的主流方法。在OO分析与设计中,我们首先从问题领域中抽象出领域模型,在领域模型中以适当的粒度归纳...

克虏伯
昨天
0
0
Coding and Paper Letter(四十)

资源整理。 1 Coding: 1.Tomislav Hengl撰写的非官方作者指南:Michael Gould•Wouter Gerritsma。 UnofficialGuide4Authors 2.R语言包rwrfhydro,社区贡献的工具箱,用于管理,分析和可视化...

胖胖雕
昨天
0
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部