xiaohema

一、admin的源码流程

首先可以确定的是:路由关系一定对应一个视图函数

a、当点击运行的时候,会先找到每一个app中的admin.py文件,并执行

b、执行urls.py 

admin.site是什么?

admin.site,urls    返回的是一个元组,里面的第一个元素是一个列表

django-admin的源码流程
我们自己生成的动态的访问url
====================================初级版=========================
from django.shortcuts import HttpResponse
from django.conf.urls import url
from django.contrib import admin
from app01 import views
def login(request):
    return HttpResponse("ok")

url_list = []

for model_class,v in admin.site._registry.items():
    print(model_class)  #打印的是每一个类<class \'app01.models.UserInfo\'>
    cls_name = model_class._meta.model_name #当前类名称的小写
    app_name = model_class._meta.app_label  #当前app的名称
    val = url(r\'^{0}/{1}/$\'.format(app_name,cls_name), login, name="login")
    url_list.append(val)

urlpatterns = [
    url(r\'^admin/\', admin.site.urls),
    # admin.site这个对象里面有一个属性_registry = {}
    #点击urls查看源码返回的是一个元组,元组的第一个元素是一个列表
    url(r\'^index/\', ([
            url(r\'^app01/userinfo/$\', login,name="login"),
             url(r\'^app01/roles/$\', login,name="login"),
        ],None,None)),
        
    url(r\'^index2/\', (url_list,None,None,)),  #吧上面定义的列表拿下来,这是后就动态生成了
]
================================升级============================
路径http://127.0.0.1:8001/index/app01/roles/后面还有增删改查的路径
http://127.0.0.1:8001/index/app01/roles/add/
http://127.0.0.1:8001/index/app01/roles/1/change/
http://127.0.0.1:8001/index/app01/roles/1/del/

实现流程
from django.shortcuts import HttpResponse
from django.conf.urls import url
from django.contrib import admin
from app01 import views
def login(request):
    return HttpResponse("ok")

def change_list(request):
    return HttpResponse("列表页面")

def add_view(request):
    return HttpResponse("添加页面")

def change_view(request,nid):
    return HttpResponse("修改页面")

def delete_view(request,nid):
    return HttpResponse("删除页面")

url_list = []

for model_class,v in admin.site._registry.items():
    print(model_class)  #打印的是每一个类<class \'app01.models.UserInfo\'>
    cls_name = model_class._meta.model_name #当前类名称的小写
    app_name = model_class._meta.app_label  #当前app的名称
    urls_list = url(r\'^{0}/{1}/$\'.format(app_name,cls_name), change_list, name="login")
    url_list.append(urls_list)

    add_url = url(r\'^{0}/{1}/add/$\'.format(app_name, cls_name), add_view, name="login")
    url_list.append(add_url)

    change_url = url(r\'^{0}/{1}/(\d+)/change/$\'.format(app_name, cls_name), change_view, name="login")
    url_list.append(change_url)

    del_url = url(r\'^{0}/{1}/(\d+)/del/$\'.format(app_name, cls_name), delete_view, name="login")
    url_list.append(del_url)

urlpatterns = [
    url(r\'^admin/\', admin.site.urls),
    # admin.site这个对象里面有一个属性_registry = {}
    #点击urls查看源码返回的是一个元组,元组的第一个元素是一个列表
    url(r\'^index/\', (
        [
            url(r\'^app01/userinfo/$\', login,name="login"),
            url(r\'^app01/roles/$\', login,name="login"),
         ],None,None)),
    url(r\'^index2/\', (url_list,None,None,)),  #吧上面定义的列表拿下来,这是后就动态生成了
]

说明了:
    url的本质:它读取_registry所有字典里面的数据,为字典里面的每一个类生成了4个url
==================================修改上面的版本=============================
定义了一个
def get_urls():
    temp = [
        url(r\'^$\'.format(app_name, cls_name), change_list),
        url(r\'^add/$\'.format(app_name, cls_name), add_view),
        url(r\'^del/$\'.format(app_name, cls_name), delete_view),
        url(r\'^change/$\'.format(app_name, cls_name), change_view)
    ]
    return temp
    
url_list = []
for model_class,v in admin.site._registry.items():
    print(\'-------\',model_class)  #打印的是每一个类<class \'app01.models.UserInfo\'>
    cls_name = model_class._meta.model_name #当前类名称的小写
    app_name = model_class._meta.app_label  #当前app的名称
    方式一:
    # all_urls = url(r\'^{0}/{1}/\'.format(app_name,cls_name), (get_urls(),None,None,))
    方式二:
    all_urls = url(r\'^{0}/{1}/\'.format(app_name,cls_name), include(get_urls()) )   
    url_list.append(all_urls)

    
urlpatterns = [
    url(r\'^admin/\', admin.site.urls),
    # admin.site这个对象里面有一个属性_registry = {}
    #点击urls查看源码返回的是一个元组,元组的第一个元素是一个列表
    url(r\'^index/\', (
        [
            url(r\'^app01/userinfo/\', ([
                                           url(r\'^$\', change_list, name="login"),
                                           url(r\'^add/$\', add_view, name="login"),
                                           url(r\'^(\d+)/del/$\', delete_view, name="login"),
                                           url(r\'^(\d+)/change/$\', change_view, name="login"),
                                       ],None,None),name="login"),
            url(r\'^app01/usertype/\', ([
                                           url(r\'^$\', change_list, name="login"),
                                           url(r\'^add/$\', add_view, name="login"),
                                           url(r\'^(\d+)/del/$\', delete_view, name="login"),
                                           url(r\'^(\d+)/change/$\', change_view, name="login"),
                                       ], None, None), name="login"),],None,None)),
            url(r\'^app02/article/\', ([
                                           url(r\'^$\', change_list, name="login"),
                                           url(r\'^add/$\', add_view, name="login"),
                                           url(r\'^(\d+)/del/$\', delete_view, name="login"),
                                           url(r\'^(\d+)/change/$\', change_view, name="login"),
                                       ],None,None),name="login"),

    # index和index2的两个是一样的,我们可以用index2的方式替代index
    url(r\'^index2/\', (url_list,None,None,)),  #吧上面定义的列表拿下来,这是后就动态生成了
]


include的本质就是:返回了一个元组,元组的第一个是这个模块
include里面
    既可以写一个列表include([]),利用include做分发
    也可以返回一个字符串:帮我们去找到这个模块,找到所有的映射关系

include(model_admin.urls)
model_admin是什么?ModelAdmin对象的urls

总结

- admin源码流程
        a. 运行程序,找到每一个app中的 admin.py 文件,并加载
            - app01.admin.py 
                - 创建admin.site中的对象
                - 执行对象的 register方法,目的:将注册类添加到 _registry中 
                    _registry = {  
                        key是传进来的model   value:是ModelAdmin的对象,传了两个参数
                        models.Role: ModelAdmin(models.Role,admin.site),
                        models.UserInfo: ModelAdmin(models.UserInfo,admin.site)
                        models.UserType: ModelAdmin(models.UserType,admin.site)
                    }
                
            - app02.admin.py
                - 用app01.admin中创建那个admin.site对象
                - 执行对象的 register方法,目的:讲注册类添加到 _registry中 
                    _registry = {
                        models.Role: ModelAdmin(models.Role,admin.site),
                        models.UserInfo: ModelAdmin(models.UserInfo,admin.site)
                        models.UserType: ModelAdmin(models.UserType,admin.site)
                        models.Article: ModelAdmin(models.Article,admin.site)
                    }
        
            admin.site是一个对象(单例模式创建),其中封装了: 
                _registry = {
                    models.Role: ModelAdmin(models.Role,admin.site),
                    models.UserInfo: ModelAdmin(models.UserInfo,admin.site)
                    models.UserType: ModelAdmin(models.UserType,admin.site)
                    models.Article: ModelAdmin(models.Article,admin.site)
                }
        b. urls.py 
            再次调用 admin.site 对象的 urls属性:
                urlpatterns = [
                    url(r\'^admin/\', admin.site.urls),
                ]
            
            class ModelAdmin(object):
                def __init__(self,model_class,site):
                    self.model_class = model_class
                    self.site = site 
                    
                def changelist_view(self,request):
                    data_list = self.model_class.objects.all()   #是动态的
                    return HttpResponse(\'列表页面\')

                def add_view(self,request):
                    return HttpResponse(\'添加页面\')


                def delete_view(self,request,nid):
                    return HttpResponse(\'删除页面\')

                def change_view(self,request,nid):
                    return HttpResponse(\'修改页面\')
                
                def get_urls(self):
                     urlpatterns = [
                        url(r\'^$\', self.changelist_view),
                        url(r\'^add/$\', self.add_view),
                        url(r\'^(.+)/delete/$\', self.delete_view),
                        url(r\'^(.+)/change/$\', self.change_view),
                    ]
                    return urlpatterns
                
                @property 
                def urls(self):
                    return self.get_urls()
                
            
            class AdminSite(object):
                def __init__(self):
                    self._registry = {}
                    
                def register(self,model_class,model_admin):
                    self._registry[model_class] = model_admin(model_class,self)
                
                def get_urls(self):
                    """
                    models.Role: ModelAdmin(models.Role,admin.site),
                    models.UserInfo: ModelAdmin(models.UserInfo,admin.site)
                    models.UserType: ModelAdmin(models.UserType,admin.site)
                    models.Article: ModelAdmin(models.Article,admin.site)
                    """
                    url_list = []
                    for model_class,model_admin in self._registry.items():
                        model_class是一个类
                        app_name = model_class._meta.app_label
                        model_name = model_class._meta.model_name 
                        url_list += [
                            url(\'%s/%s\' %(app_name,model_name,), include(model_admin.urls))
                        ]
                        
                    return url_list
                        
                
                @property
                def urls(self):
                    return (self.get_urls(), None,None )
            

 

 

 

分类:

技术点:

相关文章: