Django URL name 詳解
from django.conf.urls import url
from django.contrib import admin
from calc import views as calc_views
urlpatterns = [
url(r'^add/$', calc_views.add, name='add'),
url(r'^add/(\d+)/(\d+)/$', calc_views.add2, name='add2'),
url(r'^admin/', admin.site.urls),
]
name='add' 是用來做什么的呢
簡單來說,name 可以用於在 templates,models,views ..... 中得到對應的網址,相當於給網址取了個名字,只要這個名字不變,網址變了也能通過名字獲取到。
為了進一步弄清楚這個問題,先新建一個首頁的視圖和 url
修改 calc/views
from django.http import HttpResponse
from django.shortcuts import render
def index(request):
return render(request, 'home.html')
render 是渲染模板。
將 calc 這個 app 加入到 settings.py
INSTALLED_APPS = [
'django.contrib.admin',
'django.contrib.auth',
'django.contrib.contenttypes',
'django.contrib.sessions',
'django.contrib.messages',
'django.contrib.staticfiles',
'calc',
]
這樣,使用 render 的時候,Django 就會找到 INSTALL_APPS 中列出的各個 app 下的 templates 中的文件。
提示,DEBUG=True 的時候,django 還可以自動找到個 app 下 static 文件夾中的靜態文件(js,css,圖片等資源)
在 calc 這個 app 中新建一個 templates 文件夾,在 templates 中新建一個 home.html
<!DOCTYPE html>
<html>
<head>
<title>計算器</title>
</head>
<body>
<a href="/add/4/5/">計算 4+5</a>
</body>
</html>
修改 urls.py
urlpatterns = [
url(r'^$',calc_views.index,name='home'), # new
url(r'^add/$',calc_views.add,name='add'),
url(r'^add/(\d+)/(\d+)/$',calc_views.add2,name='add'),
url(r'^admin/', admin.site.urls),
]
我們計算加法時候用的是 /add/4/5/ ,后來需求發生變化,比如改成 4_add_5 ,但是在網頁中,代碼中很多地方都寫死的 /add/4/5/,比如模板中可能是這么寫的
<a href="add/4/5/">計算4+5</a>
如果這樣寫死,會使得在修改網址(正則)后,模板(templates),視圖(views.py,用以跳轉),模型(models.py,可以用於獲取對象對應得地址)用了此網址的,都得進行相應的更改,修改的代價很大。
有沒有更優雅的方式來解決這個問題呢
先說一下如何使用 python 代碼獲取對應得網址(可以用在 views.py,model.py 等各種需要轉換得到網址的地方 )
首先
python manage.py shell
>>> from django.urls import reverse
>>> reverse('add2',args=(21312,321321))
u'/add/21312/321321/'
>>> reverse('home')
u'/'
reverse 接收 url 中的 name 作為第一個參數,在代碼中可以通過 reverse() 來獲取對應得網址(這個網址可以用來跳轉,也可以計算相應頁面的地址),只要對應的 url 的 name 不改,就不用改戴拿中的網址。
在網頁模板中也是一樣
# 不帶參數的
{% url 'name' %}
# 帶參數的:參數可以是變量名
{% url 'name' 參數 %}
# 例如
<a href="{% url 'add2' 4 5 %}">link</a>
上面代碼渲染成的最終頁面是
<a href="add/4/5/">link</a>
這樣就可以通過 {% url 'add2' 4 5 %} 獲取到對應得網址 /add/4/5/
當 urls.py 進行更改,前提是不改 name (這個參數獲取后不要輕易改),獲取的網址也會動態的跟着變,比如改成:
url(r'^new_add/(\d+)/(\d+)\$',calc_views.add2,name='add2'),
注意看 add 變成了 new_add ,但是后面的 name = 'add2' 沒改,這時 {% url 'add2' 4 5 %} 就會渲染對應的網址成 /new_add/4/5/
用在 views.py 或 model.py 等地方的 reverse 函數,同樣會根據 name 對應的 url 獲取到新的網址。
想要改網址的時候,修改 urls.py 中的正則表達式的部分(url 參數第一部分),name 不變的前提下,其他地方不需要修改。
另外,如果用戶收藏夾中的 url 是舊的,如何讓以前的 /add/3/4/ 跳轉到現在的新網址呢
Django 不會幫你做這個,需要自己來寫一個跳轉方法:
具體思路是,在 views.py 寫一個跳轉的函數
from django.http import HttpResponseRedirect
from django.core.urlresolvers import reverse # django 1.4.x - django 1.10.x
# from django.urls import reverse # new in django 1.10.x
def old_add2_redirect(request, a, b):
return HttpResponseRedirect(
reverse('add2', args=(a, b))
)
urls.py 中
url(r'^add/(\d+)/(\d+)/$', calc_views.old_add2_redirect),
url(r'^new_add/(\d+)/(\d+)/$', calc_views.add2, name='add2'),
這樣,假如用戶收藏夾中有 /add/4/5/ ,訪問時就會自動跳轉到新的 /new_add/4/5/ 了
