Django4.0 URL調(diào)度器-URL命名空間

2022-03-16 17:42 更新

介紹

URL 命名空間允許你使用唯一的反向命名URL模式( named URL patterns ),即便不同應(yīng)用程序使用相同的 URL 名稱。對于第三方應(yīng)用程序來說,始終使用 URL 命名空間是個好習(xí)慣(像在教程里說的那樣)。同樣,如果已部署了應(yīng)用程序的多個實例,它也允許你反向解析 URL。換句話說,因為單個應(yīng)用的多個實例會分享已命名的 URL,命名空間提供了區(qū)分這些已命名 URL 的方法。

對于特定站點,正確使用URL名稱空間的Django應(yīng)用程序可以部署多次。比如 ?django.contrib.admin? 有一個 ?AdminSite ?,它允許admin實例部署多次。

URL 命名空間分為兩部分,它們都是字符串:

應(yīng)用程序命名空間

這描述了正在部署的程序名。單個應(yīng)用的每個實例擁有相同的命名空間。比如,Django admin 應(yīng)用有可預(yù)測的應(yīng)用命名空間 ?'admin'? 。

實例命名空間

這標(biāo)識了應(yīng)用程序的特定實例。實例命名空間應(yīng)該是完整項目唯一的。但是實例命名空間可以和應(yīng)用命名空間相同。這常用來指定應(yīng)用的默認(rèn)實例。比如,默認(rèn)Django admin 實例擁有名為?'admin'? 的實例命名空間。

被指定的命名空間 URL 使用 ?':'? 操作符。比如,使用 ?'admin:index'? 引用admin 應(yīng)用的首頁。這表明命名空間為 ?'admin'? ,命名 URL 為 ?'index'? 。

命名空間也可以嵌套。命名 URL ?'sports:polls:index'? 將在命名空間 ?'polls'? 中尋找命名為 ?'index'? 的模式,該模式是在頂層命名空間 ?'sports'? 中定義的。

反向命名空間URLs

當(dāng)給定一個命名空間 URL(例如 ?'polls:index'? )解析時,Django 會將完全限定的名稱拆分成多個部分,然后嘗試下面的查詢:

  • 首先,Django 查找匹配 ?application namespace? (這個例子里是 ?'polls'? )。這將產(chǎn)生應(yīng)用實例列表。
  • 如果定義了當(dāng)前應(yīng)用程序,Django 會為這個實例查找并返回 URL 解析器??梢杂?nbsp;?reverse()? 函數(shù)的 ?current_app參數(shù)來指定當(dāng)前應(yīng)用程序。url 模板標(biāo)簽使用當(dāng)前已解析的視圖命名空間當(dāng)作 ?RequestContext中的應(yīng)用程序。你可以通過設(shè)置在 ?request.current_app? 屬性上的當(dāng)前應(yīng)用來覆蓋這個默認(rèn)配置。
  • 如果當(dāng)前沒有應(yīng)用程序,Django 會尋找默認(rèn)的應(yīng)用實例。默認(rèn)應(yīng)用程序?qū)嵗蔷哂信c實例命名空間匹配的應(yīng)用程序命名空間的實例(比如, ?polls ?實例被稱為 ?'polls'? )。
  • 如果沒有默認(rèn)的應(yīng)用程序?qū)嵗?,Django 將會引用最后一次部署的應(yīng)用程序?qū)嵗瑹o論其實例命名是什么。
  • 如果提供的命名空間無法在第一步里匹配應(yīng)用程序命名空間,Django 會嘗試直接尋找命名空間來作為實例命名空間。

如果有嵌套的命名空間,則會對命名空間的每個部分重復(fù)這些步驟,直到視圖名不被解析為止,然后視圖名稱將被解析為已找到的命名空間中的一個 URL 。

例如:

為了展示這個解決策略的實際作用,請考慮教程里 ?polls應(yīng)用程序的兩個實例案例:分別被稱為 ?'author-polls'? 和 ?'publisher-polls'? 。假設(shè)我們已經(jīng)增強了這個應(yīng)用程序,以便會在創(chuàng)建和顯示 ?polls時考慮實例命名空間

from django.urls import include, path

urlpatterns = [
    path('author-polls/', include('polls.urls', namespace='author-polls')),
    path('publisher-polls/', include('polls.urls', namespace='publisher-polls')),
]
from django.urls import path

from . import views

app_name = 'polls'
urlpatterns = [
    path('', views.IndexView.as_view(), name='index'),
    path('<int:pk>/', views.DetailView.as_view(), name='detail'),
    ...
]

使用這步后,可以進(jìn)行以下查找:

如果我們在實例 ?'author-polls'? 中渲染詳情頁 - ?'polls:index'? 將解析為 ?'author-polls'? 的首頁;比如下面兩種都將觸發(fā) ?"/author-polls/"? 。

在基于類的視圖里的方法:

reverse('polls:index', current_app=self.request.resolver_match.namespace)

在模板中:

{% url 'polls:index' %}

如果我們在站點某處渲染一個頁面 - ?'polls:index'? 將被解析為 ?polls的最后一個注冊實例。因為這里沒有默認(rèn)實例( ?'polls'? 的實例命名空間),所以將使用 ?polls的最后一個注冊實例。這將是 ?'publisher-polls'? ,因為它是在 ?urlpatterns? 的最后面聲明的。

?'author-polls:index'? 會一直被解析為實例 ?'author-polls'? 的首頁(對于 ?'publisher-polls'? 同樣如此)。

如果還有一個默認(rèn)實例 - 例如,一個叫 ?'polls'? 的實例 - 唯一的變化就是沒有當(dāng)前實例(上面列表中的第二項)。在這個例子 ?'polls:index'? 將解析為默認(rèn)實例的首頁而不是在 ?urlpatterns中最后聲明的實例。

URL 命名空間和包含的 URLconfs

有兩種辦法指定包含的URLconfs應(yīng)用程序空間。

首先,你可以在包含的 URLcon 模塊中設(shè)置一個 ?app_name? 屬性,在相同層作為 ?urlpatterns屬性。你必須傳遞實際的模塊或?qū)υ撃K的一個字符串引用傳遞給 ?include()? ,而不是 ?urlpatterns ?本身的列表。

from django.urls import path

from . import views

app_name = 'polls'
urlpatterns = [
    path('', views.IndexView.as_view(), name='index'),
    path('<int:pk>/', views.DetailView.as_view(), name='detail'),
    ...
]
from django.urls import include, path

urlpatterns = [
    path('polls/', include('polls.urls')),
]

?polls.urls? 里的 URLs 定義將具有應(yīng)用程序命名空間 ?polls

其次,你可以包括一個包含嵌入式命名空間數(shù)據(jù)的對象。如果你 ?include()? 了一個 ?path()? 或 ?re_path()? 實例的列表,那個對象里包含的 URLs 將被添加到全局命名空間內(nèi)。但是,你也可以 ?include()? 一個包含以下內(nèi)容的2元組:

(<list of path()/re_path() instances>, <application namespace>)

例如:

from django.urls import include, path

from . import views

polls_patterns = ([
    path('', views.IndexView.as_view(), name='index'),
    path('<int:pk>/', views.DetailView.as_view(), name='detail'),
], 'polls')

urlpatterns = [
    path('polls/', include(polls_patterns)),
]

這會將指定的 URL 模式包含到給定的應(yīng)用程序命名空間里。

使用 ?include()? 的 ?namespace ?參數(shù)來指定實例命名空間。如果實例命名空間沒有被指定,會默認(rèn)已被導(dǎo)入的 URLconf 的應(yīng)用程序命名空間。這意味著它也將成為那個命名空間的默認(rèn)實例。


以上內(nèi)容是否對您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號
微信公眾號

編程獅公眾號