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

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数据库对象)。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,561评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,218评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,162评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,470评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,550评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,806评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,951评论 3 407
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,712评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,166评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,510评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,643评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,306评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,930评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,745评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,983评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,351评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,509评论 2 348

推荐阅读更多精彩内容