在Django中,如果我们新建一个项目,只要在admin.py文件中注册,就可以对其相应的文件进行增删改查操作。

而我们在路由系统中只看到了一条信息:url(r'^admin/', admin.site.urls),但是我们确实是可以进行增删改查操作?这其中到底有什么端倪了?

下面我们会从Django源码的角度来分析admin执行流程。

1.循环加载执行所有已经注册的app中的admin.py文件

def autodiscover():
autodiscover_modules('admin', register_to=site)

2.执行admin.py文件中的代码

class BookAdmin(admin.ModelAdmin):
list_display = ("title",'publishDate', 'price') admin.site.register(Book, BookAdmin)
admin.site.register(Publish)

3.admin.site

这里应用的是一个单例模式,对于AdminSite类的一个单例模式,执行的每一个app中的每一个admin.site都是一个对象

4.执行register方法

admin.site.register(Book, BookAdmin)
admin.site.register(Publish)

register函数源码信息:

class AdminSite(object):
......
def register(self, model_or_iterable, admin_class=None, **options): if not admin_class: #如果没有设置admin_class这个类
admin_class = ModelAdmin #那么就会执行默认的样式 if isinstance(model_or_iterable, ModelBase):
model_or_iterable = [model_or_iterable]
for model in model_or_iterable: #在名称列表中循环
......(省略)
self._registry[model] = admin_class(model, self) #这句话是关键

执行完admin.site.register()代码之后,就创建_registry这个字典,表名为键,admin_class(样式)为值,我们也可以打印这个字典。

到这里,注册结束。

5.admin的URL配置

上面已经注册成功了,现在该解释URL自动生成的过程了。

我们先来分析Django源码是如何实现的,从url(r'^admin/', admin.site.urls)出发,我们可以直接找urls函数

    @property
def urls(self):
return self.get_urls(), 'admin', self.name

随后我们去找get_urls()这个函数,

    def get_urls(self):
from django.conf.urls import url, include from django.contrib.contenttypes import views as contenttype_views def wrap(view, cacheable=False):
def wrapper(*args, **kwargs):
return self.admin_view(view, cacheable)(*args, **kwargs)
wrapper.admin_site = self
return update_wrapper(wrapper, view) # Admin-site-wide views.
urlpatterns = [
url(r'^$', wrap(self.index), name='index'),
url(r'^login/$', self.login, name='login'),
url(r'^logout/$', wrap(self.logout), name='logout'),
url(r'^password_change/$', wrap(self.password_change, cacheable=True), name='password_change'),
url(r'^password_change/done/$', wrap(self.password_change_done, cacheable=True),
name='password_change_done'),
url(r'^jsi18n/$', wrap(self.i18n_javascript, cacheable=True), name='jsi18n'),
url(r'^r/(?P<content_type_id>\d+)/(?P<object_id>.+)/$', wrap(contenttype_views.shortcut),
name='view_on_site'),
]

关于urlpatterns是如何生成的我们不了解,只知道在这里就生成了增删改查的URL,现在我们用自己的手段来实现这一过程。

6.自定义实现admin配置

首先,需要用到路由分发,就像下面这样:

def test01(request):
return HttpResponse("test01") def test02(request):
return HttpResponse("test02") def test03(request):
return HttpResponse("test03") urlpatterns = [
url(r'^admin/', admin.site.urls),
url(r'^kebi/',([
url(r'^test01/', test01),
url(r'^test02/', test02), #一条URL包含很多条内置的URL,这个部分可以封装到函数中
url(r'^test03/', test03),
],None,None))
]

里面还可以再次封装,就像这样:

def test01(request):
return HttpResponse("test01") def test0203(request):
return HttpResponse("test0203") def test0204(request):
return HttpResponse("test0204") def test05(request):
return HttpResponse("test05") urlpatterns = [
url(r'^admin/', admin.site.urls),
url(r'^kebi/',([
url(r'^test01/', test01),
url(r'^test02/', ([
url(r'^test03/', test0203),
url(r'^test04/', test0204),
],None,None)),
url(r'^test05/', test05),
],None,None))
]

我们将其封装到一个函数之中:

from django.conf.urls import url
from django.contrib import admin
from django.shortcuts import HttpResponse def test01(request):
return HttpResponse("test01") def get_urls():
urlpatterns = [] for model,model_config in admin.site._registry.items():
model_name = model._meta.model_name #model可以说是类名,现在要获取字符串拼接,获取类名,属于字符串
app_label = model._meta.app_label #app名,属于字符串
urlpatterns.append(url('%s/%s'%(app_label,model_name),test01))
return urlpatterns urlpatterns = [
url(r'^admin/', admin.site.urls),
url(r'^kebi/',(get_urls(),None,None)) #只要urlpatterns有值,就可以做分发
]

这样就生成了:

http://127.0.0.1:8000/kebi/app02/publish
http://127.0.0.1:8000/kebi/app01/authors

仅仅如此,还没有满足要求,那么就在做一次分发,就像这样:

from django.contrib import admin
from django.shortcuts import HttpResponse def show_list(request):
return HttpResponse("查看列表") def add_list(request):
return HttpResponse("增加列表") def change_list(request,id):
return HttpResponse("更改列表") def delete_list(request,id):
return HttpResponse("删除列表") def get_urls_func():
temp = []
#现在来写增删改查
temp.append(url('^$',show_list))
temp.append(url('/add/$',add_list))
temp.append(url('/(\d+)/change/$',change_list))
temp.append(url('/(\d+)/delete/$',delete_list)) return temp def get_urls():
urlpatterns = [] for model,model_config in admin.site._registry.items():
model_name = model._meta.model_name #model可以说是类名,现在要获取字符串拼接
app_label = model._meta.app_label #app名
urlpatterns.append(url('%s/%s'%(app_label,model_name),(get_urls_func(),None,None)))
return urlpatterns urlpatterns = [
url(r'^admin/', admin.site.urls),
url(r'^kebi/',(get_urls(),None,None)) #只要urlpatterns有值,就可以做分发
]

现在能够访问的有这些:

#两个表都会自动生成增删改查的URL
http://127.0.0.1:8000/kebi/app02/publish
http://127.0.0.1:8000/kebi/app02/publish/add
http://127.0.0.1:8000/kebi/app02/publish/1/change
http://127.0.0.1:8000/kebi/app02/publish/2/delete http://127.0.0.1:8000/kebi/app01/authors
http://127.0.0.1:8000/kebi/app01/authors/add
http://127.0.0.1:8000/kebi/app01/authors/1/change
http://127.0.0.1:8000/kebi/app01/authors/2/delete

下面是效果:

Django——admin源码分析的更多相关文章

  1. admin源码分析

    django settings 源码分析 导入settingso模块,进入源码,会发现settings是一个 单例 LazySettings类实例化产生的一个对象,LazySettings实例化后就会 ...

  2. $Django cbv源码分析 djangorestframework框架之APIView源码分析

    1 CBV的源码分析 #视图 class login (View): pass #路由 url(r'^books/$', views.login.as_view()) #阅读源码: #左侧工程栏--- ...

  3. django -admin 源码解析

    admin源码解析 单例模式 单例模式(Singleton Pattern)是一种常用的软件设计模式,该模式的主要目的是确保某一个类只有一个实例存在.当你希望在整个系统中,某个类只能出现一个实例时,单 ...

  4. django --- DetailView源码分析

    [背景] 最近在看django官方文档的class-based-views这一节的时候一直不得要领,感觉自己清楚,但是回想起来又没有脉络:于是没有办法只 能是“暗中观察”django的源码了. 刚打开 ...

  5. Django——Session源码分析

    首先我们导入django.contrib.sessions.middleware这个中间件,查看里面的Session源码 from django.contrib.sessions.middleware ...

  6. Django admin源码剖析

    单例模式 单例模式(Singleton Pattern)是一种常用的软件设计模式,该模式的主要目的是确保某一个类只有一个实例存在.当你希望在整个系统中,某个类只能出现一个实例时,单例对象就能派上用场. ...

  7. Django之REST framework源码分析

    前言: Django REST framework,是1个基于Django搭建 REST风格API的框架: 1.什么是API呢? API就是访问即可获取数据的url地址,下面是一个最简单的 Djang ...

  8. django Rest Framework----APIView 执行流程 APIView 源码分析

    在django—CBV源码分析中,我们是分析的from django.views import View下的执行流程,这篇博客我们介绍django Rest Framework下的APIView的源码 ...

  9. django---APIView源码分析

    django---APIView源码分析 前言:APIView基于View 看这部分内容一定要懂django-CBV里的内容 在django-CBV源码分析中,我们是分析的from django.vi ...

随机推荐

  1. 219. Contains Duplicate II【easy】

    219. Contains Duplicate II[easy] Given an array of integers and an integer k, find out whether there ...

  2. centos7下安装mysql5.7和jdk 1.8

    安装mysql5.7 具体安装过程可参见官网:A Quick Guide to Using the MySQL Yum Repository 进入/usr/local/src文件夹. cd /usr/ ...

  3. C++函数指针和类成员函数指针

    一.函数指针——基本形式 char Fun(int n) { return char(n); } //char(*pFun)(int); void main() { char(*pFun)(int); ...

  4. ZOJ 3963 Heap Partition set维护。给一个序列,将其划分成尽量少的序列,使每一个序列满足按照顺序构造二叉树,父母的值<=孩子的值。

    Heap Partition Time Limit: Seconds Memory Limit: KB Special Judge A sequence S = {s1, s2, ..., sn} i ...

  5. 【Mac + Python3.6 + ATX基于facebook-wda】之IOS自动化(一):WebDriverAgent安装

    此篇介绍如何安装WebDriverAgent,下一篇介绍facebook-wda库的安装使用以及自动化脚本的开发. 前言: 对于iOS的应用的测试,如果不需要用到图像识别,推荐使用这个项目facebo ...

  6. 【转】Monkey测试2——Monkey测试策略

    Monkey的测试策略 一. 分类 Monkey测试针对不同的对象和不同的目的采用不同的测试方案,首先测试的对象.目的及类型如下: 测试的类型分为:应用程序的稳定性测试和压力测试 测试对象分为:单一a ...

  7. Eclipse 创建 Java 接口

    打开新建 Java 接口向导 新建 Java 接口向导可以创建新的 Java 接口.打开向导的方式有: 点击 File 菜单并选择 New > Interface 在 Package Explo ...

  8. Unity3D学习笔记——Android重力感应控制小球

    一:准备资源 两张贴图:地图和小球贴图. 二:导入资源 在Assets下建立resources文件夹,然后将贴图导入. 三:建立场景游戏对象 1.建立灯光: 2.创建一个相机,配置默认. 3.建立一个 ...

  9. java前端传入的json字符串保存到表中的方法

    表 service_goods_base 字段如下: 传入的json 字符串: servicePictureArray  :  [{"picServiceUrl": "h ...

  10. Notepad++ 64位 插件管理

    notepad++ 64bit 没有插件管理,如何添加呢? 1.访问https://github.com/bruderstein/nppPluginManager/releases,下载  Plugi ...