一、URL路由基礎
URL是web服務的路口,用戶通過瀏覽器發送過來的任何請求都會被發送到一個指定的URL地址里,然后被響應。
在django項目中編寫路由就是向外暴露我們接收哪些URL的請求,除此之外任何的URL都不會被處理,URL路由就是web服務對外暴露的API
二、Django處理請求
確定要使用的 URLconf 模塊,通常是settings中 ROOT_URLCONF 設置的值,如果傳入的 HttpRequest 對象具有 urlconf 屬性(中間件設置),則使用其值代替settings中 ROOT_URLCONF
Django加載模塊并查找可用的 urlpatterns ,它是 django.conf.urls.url() 實例的一個列表
按順序運行每個URL模式,匹配成功就停下來,所以 順序很關鍵
匹配成功導入給定的視圖,它是一個python函數,或基于類的視圖,視圖將獲得如下參數
如果沒有URL模式匹配,或者過程出錯了,將調用錯誤處理視圖
三、簡單的路由配置
from django.conf.urls import url
urlpatterns=[ url(正則表達式,view視圖函數,參數,別名)]
示例的URLconf:
from django.urls import urlfrom . import viewsurlpatterns = [ url(r'^articles/2003/$', views.special_case_2003), url(r'^articles/([0-9]{4})/$', views.year_archive), url(r'^articles/([0-9]{4})/([0-9]{2})/$', views.month_archive), url(r'^articles/([0-9]{4})/([0-9]{2})/([0-9]+)/$', views.article_detail),]注:
請求的例子及匹配的url
/articles/2005/03/將匹配列表中的第三個模式。Django將調用函數views.month_archive(request, '2005', '03')。
/articles/2005/3/不匹配任何URL模式,因為列表中的第三個模式要求月份是兩個數字。
/articles/2003/將匹配列表中的第一個模式不是第二個,因為模式按順序從上往下匹配,第一個會首先被匹配。Django會調用函數views.special_case_2003(request)
/articles/2003不匹配任何一個模式,因為每個模式都要求URL以一個斜杠結尾。
/articles/2003/03/03/將匹配最后一個模式。Django將調用函數views.article_detail(request, '2003', '03', '03')。
是否開啟URL訪問地址后面 不為/跳轉至帶有/路徑的配置項
APPEND_SLASH=True
Django settings.py配置文件中默認沒有 APPEND_SLASH 這個參數,但 Django 默認這個參數為 APPEND_SLASH = True。 其作用就是自動在網址結尾加'/'。
四、有名分組
有名分組的語法是 (?P<name>pattern) ,其中name是組的名字,pattern是匹配的模式
使用有名分組重寫上面的URLconf:
from django.conf.urls import urlfrom . import viewsurlpatterns = [ url(r'^articles/2003/$', views.special_case_2003), url(r'^articles/(?P<year>[0-9]{4})/$', views.year_archive), url(r'^articles/(?P<year>[0-9]{4})/(?P<month>[0-9]{2})/$', views.month_archive), url(r'^articles/(?P<year>[0-9]{4})/(?P<month>[0-9]{2})/(?P<day>[0-9]{2})/$', views.article_detail),]注:捕獲的值作為關鍵字參數而不是位置參數傳遞給視圖函數。
/articles/2005/03/ 請求將調用 views.month_archive(request, year='2005', month='03') 函數,而不是 views.month_archive(request, '2005', '03') 。
五、無名分組有名分組總結
1.無名分組
按位置傳參
分組之后,將分組好的數據當做位置傳參到視圖函數,所以視圖函數需要定義形參
示例:
? url: (r'^articles/([0-9]{4})/([0-9]{2})$', views.article_detail)
? 視圖函數: def article_detail(request,*args)
2.有名分組
按關鍵字傳參
分組后,會把分組出來的數據當做關鍵字參數傳到視圖函數,所以視圖函數需要定義形參,形參名字和分組的名字相對應,與順序無關
示例:
? url: (r'^articles/(?P<year>[0-9]{4})/(?P<mounth>[0-9]{2})/$', views.article_detail),
? 視圖函數: def article_detail(request,mounth,year)
注:有名分組和無名分組最好不要混用
六、反向解析
在django項目中,一個常見的需求是獲得URL的最終形式,以用于嵌入到生成的內容中(視圖中和顯示給用戶的URL等)或者用于處理服務器端的導航(重定向)。不希望通過硬編碼URL
Django提供了一種解決方案,只需在URL中提供一個name參數,并賦值一個你自定義的、好記的、直觀的字符串。
示例:
url配置:
模板層:
視圖層:
from django.shortcuts import reverse
七、路由分發
在每個app里各自創建一個urls.py路由模塊,然后從根路由出發,將app所屬的url請求全部轉發到相應的urls.py模塊中。
Django1.1版本的分發
from django.conf.urls import url,include
例子:
總路由:
各自路由配置url
路由分發使用的是include()方法,需要提前導入,他的參數是轉發目的地地路徑的字符串。
重點:總路由后面不能加 $
兩個不同的app,在各自的urlconf中為某一條url取了相同的name,這就會帶來麻煩。為了解決這個問題,又引出了下面的命名空間。
八、命名空間
由于name沒有作用域,Django在反解URL時,會在項目全局順序搜索,當查找到第一個name指定URL時,立即返回。URL命名空間可以保證反查到唯一的URL,即使不同的app使用相同的URL名稱。
示例:
urls.py
url(r'^blog/',include('blog.urls')),url(r'^app01/',include('app01.urls')),blog的urls.py
url(r'^blogtest/$', views.test,name='test'),
app01的urls.py
url(r'^publish/$', views.publish,name='test'),
blog的視圖函數
def test(request): url=reverse('test') return HttpResponse('blog test)app01的視圖函數
def test(request): url=reverse('test') return HttpResponse('app01 test)無論如何找index都是找的app01的index。
解決方法:在總路由分發的時候指定名稱空間,實現命名空間的做法很簡單,在urlconf文件中添加 namespace='xxx' 即可。
url(r'^blog/',include('blog.urls',namespace='blog')),url(r'^app01/',include('app01.urls',namespace='app01')),在視圖函數反向解析的時候,指定是哪個名稱空間下的
url = reverse('blog:test')在模板里也指定是
{% url 'blog:test'%}不是很推薦使用名稱空間,推薦的是在子路由的name中加入app的前綴
url(r'^publish/$', views.publish,name='app01_test'),
九、偽靜態
和真靜態URL類似。他是通過偽靜態規則把動態URL偽裝成靜態網址。
在urls.py文件中自己添加匹配 .html
url(r'^book/(?P<id>/d+.html)',views.book),
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VEVB武林網。
新聞熱點
疑難解答