文档章节

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
博文 2634
码字总数 82983
作品 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
newrelic python agent 源码分析-1

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

goodspeed
05/16
0
0
Spring MVC 原理探秘 - 容器的创建过程

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

coolblog.xyz
07/03
0
0
Django 结构及处理流程分析

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

岭南六少
2011/08/08
0
0
Django源码中的metaclass使用是如何兼容Python2和Python3的

Django源码中的metaclass使用是如何兼容Python2和Python3的 the5fire的技术博客2017-07-2814 阅读 源码metaclasspython3Django 之前看Django源码时没太注意metaclass是怎么做的2跟3的兼容,直...

the5fire的技术博客
2017/07/28
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

HTML显示json字符串并且进行格式化

通过pre标签进行格式化展示,使用JSON.stringify()方法转换。 代码如下:   <html> <head> <title>HTML显示json字符串并且进行格式化</title> </head> <body> <p id="show_p">{ "name": "B......

writeademo
1分钟前
0
0
LNMP——php-fpm

php-fpm的pool •vim /usr/local/php/etc/php-fpm.conf //在[global]部分增加include并删除我们之前www池子 • include = etc/php-fpm.d/*.conf • mkdir /usr/local/php/etc/php-fpm.d/ • ......

chencheng-linux
5分钟前
0
0
fixed在微信下的BUG

最近在一个项目中,碰到了一个问题,是属于微信内部的问题,或者说,是属于APP内部的问题,它的根源来自于fixed定位以及-webkit-overflow-scrolling的组合使用,虽然现在还无法理解问题的根源...

Jack088
9分钟前
0
0
sentry错误日志收集

安装方式 : 1 docker 2 Python 安装docker,使用命令 sudo yum -y install docker-io 启动docker 使用命令service docker start docker换源sudo curl -sSL https://get.daocloud.io/daotoo......

猿神出窍
10分钟前
0
0
input输入框的限制

1、键入时候保存数字, 问题:多个输入框的时候,连续输入非数字,导致删除的非数字出现 onkeyup="this.value=this.value.replace(/\D/g,'')" 2、onkeypress,返回true和false来设置是否键入...

大美琴
13分钟前
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部