1.什麼是路由層 簡單來說,就是通過路由層中的path函數,告訴django遇到那個url,執行那個視圖函數 2.路由層的請求流程 1.客戶在瀏覽器輸入網址→請求進入django的setting.py中的ROOT_URLCONF尋找指定使用的urls.py文件位置(如果中間件有路由功能,urls文件 ...
1.什麼是路由層
簡單來說,就是通過路由層中的path函數,告訴django遇到那個url,執行那個視圖函數
2.路由層的請求流程
1.客戶在瀏覽器輸入網址→請求進入django的setting.py中的ROOT_URLCONF尋找指定使用的urls.py文件位置(如果中間件有路由功能,urls文件功能會被其替代)
2.Django會先匹配項目目錄下的path路徑然後在匹配應用裡面的url路徑,從上至下查找url匹配,一旦匹配成功,停止查找。
3.一旦匹配url成功後,路由層向視圖層傳遞下列參數:
1)一個HttpRequest實例,即request
2)如果url中的可變參數沒有命名,那麼就按照位置來與視圖函數中的形參一一對應,如果有則按關鍵字方式傳參
4.如果匹配不到url,Django會發出異常
re_path(r'^test/$',views.test)
註意:path路徑匹配是會自動在url後面加上一個斜杠,即先匹配test沒有找到,會自動加個/在匹配一次!
如何取消自動加斜杠?
在setting.py里添加APPEND_SLASH = False 預設是True
3.路由層的格式和參數說明
3.1有名分組和無名分組
分組:就是給某一段正則表達式用小括弧擴起來
無名分組:就是將括弧內的正則表達式匹配到的內容當做位置參數傳遞給後面的視圖函數
from django.urls import path, re_path
re_path(r'^test/(\d+)/',views.test)
def test(request,xx): # xx這個參數就是對應path裡面的(\d+)
pass
有名分組:可以給正則表達式起個別名
from django.urls import path, re_path
re_path(r‘^(?P<自定義參數名>正則表達式)/$',view.xxx,name),
def test(request,自定義參數名):
pass
有名無名不能混用!單個分組可以使用多次!
3.2url反向解析
反向解析的步驟:
# 1.先給改url起一個別名
re_path(r'^test/$',views.test,name='ooo')
# 2.後端反向解析(視圖層)
先導入reverse
from django.shortcuts import reverse
print(reverse(‘ooo’)) # 得到的就是對應別名的url
# 3.前端反向解析
在需要url的地方,改成如下格式:
<a href='{% url 'ooo' %}'>111</a>
3.3有名無名的反向解析
1.無名分組
後端部分:
先導入reverse
from django.shortcuts import reverse
print(reverse(‘ooo’),args=(1,))#需要傳一個對應參數args,裡面的1可以配正則表達式匹配到
前端部分:
<a href='{% url 'ooo' 123 %}'>111</a>
# 123是匹配到url的正則表達式
2.有名分組
其實是和無名分組一樣的,但是也可以寫成關鍵字方式傳參
print(reverse(‘ooo’),kwargs={‘year’:1111}))
<a href='{% url 'ooo' year=1111a>
傳入的數字參數一般是數據的主鍵值!!
3.4路由分發
起因:分散式路由是基於django應用誕生的,有了應用則就應該有分散式路由,主路由不用在處理用戶具體路由了,而是轉而做請求的分發,具體的請求由各自的應用負責。
主路由語法:
from django.urls import path,include
# 匹配http://127.0.0.1:8000/music/index--music應用path('music/', include('music.urls'))
include('app名字.url模塊名')
作用是將當前path裡面music路由轉到include裡面的這個應用路由里處理
子路由語法:
手動創建一個urls.py,結構和主路由完全一樣
只是path後面的第一個參數,只需管主路由path匹配的後面一部分
3.5名稱空間
在各路由起別名的時候,由於我們採用的是分散式路由開發,很有可能每個開發人員起的別名是一樣的,這樣我們在進行反向解析的時候就沒有用了!
因此,名稱空間的使用很好的解決了這一問題!具體的語法如下:
# 總路由
re_path(r'^app01/',include('app01.urls',namespace='app01')),
re_path(r'^app02/',include('app02.urls',namespace='app02'))
# 解析的時候
# app01
urlpatterns = [
url(r'^reg/',views.reg,name='reg')
]
# app02
urlpatterns = [
url(r'^reg/',views.reg,name='reg')
]
# 視圖層:
reverse('app01:reg')
reverse('app02:reg')
# 模板層:
{% url 'app01:reg' %}
{% url 'app02:reg' %}
但其實,我們在起別名的時候只要保證別名不衝突,就沒有必要使用名稱空間。
一般情況下,有多個app的時候我們在起別名的時候會加上app的首碼,如name=‘app01_xxx’,這樣就能解決了!
3.6路由層的轉換器
path('page/<int:num>', views.test),
說明:1.path轉換器:可以用來批量匹配一百個網頁分別為page/1、page/2...等
2.語法為:<轉換器類型:自定義名>
3.作用:若轉換器類型匹配到對應類型的數據,則將數據按照關鍵字傳參的方式傳遞給視圖函數
4.轉換器的類型:str:匹配除了'/'之外的非空字元串;int:匹配0或者任何正整數;# slug:匹配任意由ascll字母或數字以及連字元和下劃線組成的短標簽;path:匹配非空欄位。