Django-admin組件

来源:https://www.cnblogs.com/yangqian007/archive/2018/08/28/9551710.html
-Advertisement-
Play Games

<!--done--> 知識預覽 Django 如何使用admin組件來對後臺數據進行管理的? Django admin如何實現後臺數據管理的?(admin源碼解析) 如何仿照admin實現一個自定義的增刪改查的組件? 回到頂部 Django 如何使用admin組件來對後臺數據進行管理的? 在每個a ...


知識預覽

回到頂部

Django 如何使用admin組件來對後臺數據進行管理的?

在每個app下的admin.py文件中進行註冊:
    from app名.models import 模型類名
    from django.contrib import admin
    
    admin.site.register(模型類名)
    
    class 自定義配置類名(admin.ModelAdmin):
        list_display = ["欄位名1","欄位名1"]
        
    admin.site.register(模型類名,自定義配置類名)

 

回到頂部

Django admin如何實現後臺數據管理的?(admin源碼解析)

一、啟動
    django啟動後,會載入settings中的INSTALLED_APPS
        from django.contrib import admin
    在__init.py中:
        from django.contrib.admin.sites import AdminSite, site    // 導入模塊的實例化對象,此為單例模式

        from django.utils.module_loading import autodiscover_modules
        def autodiscover():
            autodiscover_modules('admin', register_to=site)    //載入每一個app下的admin.py文件
    # 優先執行哪個app下的admin.py 是由settings中的INSTALLED_APPS的載入順序決定的
二、註冊
    admin.site.register(Author)
    
    class BookConfig(admin.ModelAdmin):
       pass
    admin.site.register(Book,BookConfig)
    
    admin.py---sites.py中:
        class AdminSite(object):
            def __init__(self):
                self._registry = {} 
                
            def register(self,model,admin_class=None):
                if not admin_class:
                    admin_class = ModelAdmin
                    
                self._registry[model] = admin_class(model, self)        
      
        site = AdminSite()
        
        admin.site._registry --> {
            <class 'django.contrib.auth.models.Group'>:<django.contrib.auth.admin.GroupAdmin object at 0x0000000003E70DD8>,
            <class 'django.contrib.auth.models.User'>:<django.contrib.auth.admin.UserAdmin object at 0x0000000003E9DCC0>,
            <class 'app01.models.Author'>:ModelAdmin(Author),
            <class 'app01.models.AuthorDetail'>:ModelAdmin(AuthorDetail),
            <class 'app01.models.Book'>: BookConfig(Book),
            <class 'app01.models.Publish'>:PublishConfig(Publish),
        }
        在app下admin.py文件中註冊,admin.site._registry中才會有此鍵值對。
三、設計url
##### 一級url分發 #####
urlpatterns = [
    
    url(r"^my_admin/",([
                        url(r"^book/",app01_views.list_book),
                        url(r"^publish/",app01_views.list_publish),
                        url(r"^food/",app02_views.list_food),
                       ],None,None))
]


##### 二級url分發 #####
urlpatterns = [
    
    url(r"^my_admin/",([
                        url(r"^book/",([
                                        url(r"^$",app01_views.list_book),
                                        url(r"^add/$",app01_views.add_book),
                                        url(r"^(\d+)/change/$",app01_views.edit_book),
                                        url(r"^(\d+)/delete/$",app01_views.del_book),
                                       ],None,None)),
                        url(r"^publish/",app01_views.list_publish),
                        url(r"^food/",app02_views.list_food),
                       ],None,None))
]

在urls.py中:
    from django.conf.urls import url
    from django.contrib import admin
    from django.shortcuts import HttpResponse
    def listview(request):
    return HttpResponse("listview")


    def addview(request):
        return HttpResponse("addview")


    def changeview(request,id):
        return HttpResponse("changeview")


    def deleteview(request,id):
        return HttpResponse("deleteview")


    def get_urls_02():
        res = [
            url(r"^$",listview),
            url(r"^add/$",addview),
            url(r"^(\d+)/change/$",changeview),
            url(r"^(\d+)/delete/$",deleteview),
        ]
        return res


    def get_urls_01():
        res = []
        print("urls.py文件中admin.site._registry -->",admin.site._registry)
        """
        urls.py文件中admin.site._registry --> {
        <class 'django.contrib.auth.models.Group'>: <django.contrib.auth.admin.GroupAdmin object at 0x0000000003E70E10>, 
        <class 'django.contrib.auth.models.User'>: <django.contrib.auth.admin.UserAdmin object at 0x0000000003E9DCF8>, 
        <class 'app01.models.Author'>: <django.contrib.admin.options.ModelAdmin object at 0x0000000003EB0940>, 
        <class 'app01.models.AuthorDetail'>: <django.contrib.admin.options.ModelAdmin object at 0x0000000003EB0978>, 
        <class 'app01.models.Book'>: <app01.admin.BookConfig object at 0x0000000003EB09B0>, 
        <class 'app01.models.Publish'>: <app01.admin.PublishConfig object at 0x0000000003EB09E8>, 
        <class 'app02.models.Food'>: <django.contrib.admin.options.ModelAdmin object at 0x0000000003EB07F0>
        }
        """
        for model,config_obj in admin.site._registry.items():
            print("模型類變數model-->",model)
            print("配置類對象config_obj-->",config_obj)

            model_name = model._meta.model_name # 獲取字元串形式的模型類名
            app_label = model._meta.app_label # 獲取字元串形式的app名
            print("model_name-->{},type(model_name)-->{}".format(model_name,type(model_name)))
            print("app_label-->{},type(model_name)-->{}".format(app_label,type(app_label)))
            """
            模型類變數model--> <class 'django.contrib.auth.models.Group'>
            配置類對象config_obj--> auth.GroupAdmin
            model_name-->group,type(model_name)--><class 'str'>
            app_label-->auth,type(model_name)--><class 'str'>
            模型類變數model--> <class 'django.contrib.auth.models.User'>
            配置類對象config_obj--> auth.UserAdmin
            model_name-->user,type(model_name)--><class 'str'>
            app_label-->auth,type(model_name)--><class 'str'>
            模型類變數model--> <class 'app01.models.Author'>
            配置類對象config_obj--> app01.ModelAdmin
            model_name-->author,type(model_name)--><class 'str'>
            app_label-->app01,type(model_name)--><class 'str'>
            模型類變數model--> <class 'app01.models.AuthorDetail'>
            配置類對象config_obj--> app01.ModelAdmin
            model_name-->authordetail,type(model_name)--><class 'str'>
            app_label-->app01,type(model_name)--><class 'str'>
            模型類變數model--> <class 'app01.models.Book'>
            配置類對象config_obj--> app01.BookConfig
            model_name-->book,type(model_name)--><class 'str'>
            app_label-->app01,type(model_name)--><class 'str'>
            模型類變數model--> <class 'app01.models.Publish'>
            配置類對象config_obj--> app01.PublishConfig
            model_name-->publish,type(model_name)--><class 'str'>
            app_label-->app01,type(model_name)--><class 'str'>
            模型類變數model--> <class 'app02.models.Food'>
            配置類對象config_obj--> app02.ModelAdmin
            model_name-->food,type(model_name)--><class 'str'>
            app_label-->app02,type(model_name)--><class 'str'>
            """
            add_url = url(r"^{}/{}/".format(app_label,model_name),(get_urls_02(),None,None))
            res.append(add_url)
        return res


    urlpatterns = [
        url(r"^my_admin/",(get_urls_01(),None,None)),
    ]

 

回到頂部

如何仿照admin實現一個自定義的增刪改查的組件?

一、啟動
    1、創建一個與Django項目無關的,可以單獨分離出來用在多個項目上的名稱為my_admin的app:
        python manage.py startapp my_admin    
    2、創建兩個與Django項目有關的兩個app:
        python manage.py startapp app01
        python manage.py startapp app02
    3、在settings.py中的INSTALLED_APPS變數中添加:
        'app01.apps.App01Config',
        'app02.apps.App02Config',
        'my_admin.apps.MyAdminConfig',
    4、將my_admin、app01和app02中的admin.py文件全部刪除,重新分別在app01和app02中添加myAdmin.py
    5、app01下models.py中添加Book,Publish,AuthorDetail,Author類
    6、app02下models.py中添加Food類
    7、遷移資料庫:
        python manage.py makemigrations
        python manage.py migrate
    8、my_admin的app下有一個apps.py文件,在此文件中添加:
        from django.utils.module_loading import autodiscover_modules
        class MyAdminConfig(AppConfig):
            name = 'my_admin'

            def ready(self):
                autodiscover_modules("myAdmin")
二、註冊
    1、my_admin的app下創建一個python package的包,名稱為service
    2、在service文件夾下新建一個sites.py文件
    3、sites.py中添加以下代碼:
        class ModelMyAdmin():
            list_display = []
            
            def __init__(self,model):
                self.model = model
            
        
        class MyAdminSite():
            def __init__(self):
                self._registry = {}
                
            def register(self,model,my_admin_class = None):
                if not my_admin_class:
                    my_admin_class = ModelMyAdmin
                self._registry[model] = my_admin_class(model)
        
        
        site = MyAdminSite()
    4、在app01下的myAdmin.py中註冊模型類:
        from my_admin.service.sites import ModelMyAdmin,site
        from app01.models import Book,Publish,Author,AuthorDetail
        
        class BookConfig(ModelMyAdmin):
            list_display = ["title","publish_date","price"]
            
            
        site.register(Book,BookConfig)
        site.register(Publish)
        site.register(Author)
        site.register(AuthorDetail)
        
        print("app01下的site._registry-->",site._registry)
        啟動項目後,列印出此字典證明已經註冊成功
        {
            <class 'app01.models.Author'>: <my_admin.service.sites.ModelMyAdmin object at 0x0000000003EA70B8>, 
            <class 'app01.models.AuthorDetail'>: <my_admin.service.sites.ModelMyAdmin object at 0x0000000003EB1B00>, 
            <class 'app01.models.Book'>: <app01.myAdmin.BookConfig object at 0x0000000003EB1EB8>, 
            <class 'app01.models.Publish'>: <my_admin.service.sites.ModelMyAdmin object at 0x0000000003EB1EF0>
        }
三、設計url
    1、在urls.py文件中:
        from django.conf.urls import url
        from my_admin.service.sites import site
        
        urlpatterns = [
            url(r'^my_admin/',site.urls),
        ]
        
    2、在sites.py文件中的MyAdminSite類中繼續添加一個urls方法:
        from django.conf.urls import url
        def get_urls_01(self):
            res = []
            for model,config_obj in self._registry.items():
                model_name = model._meta.model_name
                app_label = model._meta.app_label
                add_url = url(r'{}/{}/'.format(app_label,model_name),config_obj.urls)    #config_obj:某個model的配置類(自定義配置類或者預設配置類)對象     
                res.append(add_url)
            return res
        
        @property
        def urls(self):
            return self.get_urls_01(),None,None
            
    3、在sites.py文件中的ModelMyAdmin類中繼續添加一個urls方法:
        from django.shortcuts import render
        def listview(self,request):
            print("self-->",self) # 當前訪問模型類的配置類對象
            print("self.model-->",self.model) # 當前訪問模型類
            
            data = self.model.objects.all()
            return render(request,"listview.html",{"data_list":data})    
            
        def addview(self,request):
            return HttpResponse("addview")

        def changeview(self,request, id):
            return HttpResponse("changeview")

        def deleteview(self,request, id):
            return HttpResponse("deleteview")    
            
        def get_urls_02(self):
            res = [
                url(r'^$',self.listview)
                url(r'^add/$',self.addview)
                url(r'^(\d+)/change/$',self.changeview)
                url(r'^(\d+)/delete/$',self.deleteview
            ]
            return res
        
        @property        
        def urls(self):
            return self.get_urls_02(),None,None
為什麼要將get_urls_02的方法寫入到ModelMyAdmin類中,而不寫在MyAdminSite類中?
    將get_urls_02寫入到MyAdminSite類中,由於單例模式造成返回的是同一個頁面,如果是簡單的返回一個HttpResponse對象,是可以的;
    但是現實需求是不同的表要展示不同的視圖數據而且不同的表要有不同的配置信息,故需要寫入在ModelMyAdmin類中。



您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在看這篇文章之前,本人建議你先看看其他的關於這個問題的本質以及解決的方案。 我們做開發,通過SVN同步項目,連接的是遠程資料庫,同學在SVN導出的項目在Windows上可以運行,我在Windows上也可以運行,但是在Linux上就是不能運行--意思就是說什麼賬號,密碼,埠,許可權問題全都沒有。 但是 ...
  • OutOfMemoryError 問題相信很多朋友都遇到過,相對於常見的業務異常(數組越界、空指針等)來說這類問題是很難定位和解決的。 本文以最近碰到的一次線上記憶體溢出的定位、解決問題的方式展開;希望能對碰到類似問題的同學帶來思路和幫助。 ...
  • 1、基本類型包裝類 2、System類 3、Math類 4、Arrays類 5、大數據運算 ...
  • 有人讓我寫一下關於數據挖掘在金融方面的應用,再加上現在金融對數據方面的要求不斷提高,準備用兩篇隨筆來做個入門總結。 首先,在看這篇隨筆以前稍微補充一點金融方面的知識,因為我不是金融專業的,以下補充知識來自互聯網與個人整理,歡迎批評指正並補充說明。 1 先來瞭解一下什麼是金融市場呢? 通常狹義的金融市 ...
  • 變數定義 1.基礎定義 變數類型在變數名後 2.定義並賦值 3.類型推導 不用定義變數類型 4.簡寫(只能在函數內) 用":="代替"var" 5.定義多個變數並賦值 測試代碼 ...
  • 詳細教程 參考struts教程https://www.w3cschool.cn/struts_2/struts_configuration.html Struts2 基於MVC設計模式的web應用程式框架,它不僅僅是Struts1 的升級版本,更是一個全新的Struts架構。最初,是以WebWork ...
  • 一.文件的打開,open函數 打開模式有很多種 1. 'r': 以只讀方式打開文件。文件的指針將會放在文件的開頭。這是預設模式。 2. 'r+': 打開一個文件用於讀寫。文件指針將會放在文件的開頭,但寫入內容會寫到文件內容末尾。 3. 'w': 打開一個文件只用於寫入。如果該文件已存在則打開文件,並 ...
  • 1.什麼是servlet? Servlet(Servlet Applet),全稱Java Servlet,是用Java編寫的伺服器端程式。而這些Servlet都要實現Servlet這個介面。其主要功能在於互動式的瀏覽和修改數據,生成動態Web內容。Servlet運行於支持Java的應用伺服器中。 2 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...