Django URL name详解

1. 打开 zqxt_views/urls.pycss

1
2
3
4
5
6
7
8
9
10
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),
]

 

url(r'^add/$', calc_views.add, name='add'), 这里的 name='add' 是用来干什么的呢?html

简单说,name 能够用于在 templates, models, views ……中获得对应的网址,至关于“给网址取了个名字”,只要这个名字不变,网址变了也能经过名字获取到。python

 

为了进一步弄清这个问题,咱们先建一个首页的视图和url正则表达式

2. 修改 calc/views.pyshell

1
2
3
4
5
6
7
8
9
from  django.http  import  HttpResponse
from  django.shortcuts  import  render
 
 
def  index(request):
     return  render(request,  'home.html' )
 
 
...此处省去一些代码

render 是渲染模板,不懂先照着打就好。django

 

3. 将 'calc' 这个 app 加入到 zqxt_views/settings.py 中bash

1
2
3
4
5
6
7
8
9
10
INSTALLED_APPS  =  [
     'django.contrib.admin' ,
     'django.contrib.auth' ,
     'django.contrib.contenttypes' ,
     'django.contrib.sessions' ,
     'django.contrib.messages' ,
     'django.contrib.staticfiles' ,
 
     'calc' ,
]

这样,使用render的时候,Django 会自动找到 INSTALLED_APPS 中列出的各个 app 下的 templates 中的文件。服务器

小提示,DEBUG=True 的时候,Django 还能够自动找到 各 app 下 static 文件夹中的静态文件(js,css,图片等资源),方便开发,后面有专门的章节会讲这些。session

 

4. 咱们在 calc 这个 app 中新建一个 templates 文件夹,在templates中新建一个 home.html (关于模板更详细的能够稍后看下一节)app

文件 calc/templates/home.html 中写入如下内容(保存时用 utf8 编码

1
2
3
4
5
6
7
8
9
10
11
<!DOCTYPE html>
< html >
< head >
     < title >自强学堂</ title >
</ head >
< body >
 
< a  href = "/add/4/5/" >计算 4+5</ a >
 
</ body >
</ html >

 

修改 zqxt_views/urls.py

1
2
3
4
5
6
7
8
...此处省去一些代码
 
urlpatterns  =  [
     url(r '^$' , calc_views.index, name = 'home' ),
     url(r '^add/$' , calc_views.add, name = 'add' ),
     url(r '^add/(\d+)/(\d+)/$' , calc_views.add2, name = 'add2' ),
     url(r '^admin/' , admin.site.urls),
]

运行开发服务器,咱们访问 http://127.0.0.1:8000/ 能够看到

 

django-url-name-1.png

咱们计算加法的时候用的是 /add/4/5/ ,后来需求发生变化,好比改为 /4_add_5/,但在网页中,代码中不少地方都写死的 /add/4/5/,好比模板中多是这么写的 

1
< a  href = "/add/4/5/" >计算 4+5</ a >

若是这样写“死网址”,会使得在改了网址(正则)后,模板(template),视图(views.py,好比用于URL跳转),模型(models.py,获取记录访问地址等)用了此网址的,都必须进行相应的更改,修改的代价很大,一不当心,有的地方没改过来,就不能用了

那么有没有更优雅的方式来解决这个问题呢?固然答案是确定的。

咱们先说一下如何用 Python 代码获取对应的网址(能够用在 views.py,models.py等各类须要转换获得网址的地方):

咱们在终端上输入(推荐安装 bpython, 这样Django会用 bpython的 shell)

1
python manage.py shell

 

1
2
3
4
5
6
7
8
>>>  from  django.core.urlresolvers  import  reverse   # django 1.4.x - django 1.10.x
或者
>>>  from  django.urls  import  reverse   # Django 1.10.x - Django 2.x 新的,更加规范了
 
>>> reverse( 'add2' , args = ( 4 , 5 ))
u '/add/4/5/'
>>> reverse( 'add2' , args = ( 444 , 555 ))
u '/add/444/555/'

reverse 接收 url 中的 name 做为第一个参数,咱们在代码中就能够经过 reverse() 来获取对应的网址(这个网址能够用来跳转,也能够用来计算相关页面的地址),只要对应的 url 的name不改,就不用改代码中的网址。

在网页模板中也是同样,能够很方便的使用。

1
2
3
4
5
6
7
不带参数的:
{% url 'name' %}
带参数的:参数能够是变量名
{% url 'name' 参数 %}
 
例如:
< a  href = "{% url 'add2' 4 5 %}" >link</ a >

上面的代码渲染成最终的页面是

1
< a  href = "/add/4/5/" >link</ a >

这样就能够经过 {% url 'add2' 4 5 %} 获取到对应的网址 /add/4/5/

 

当 urls.py 进行更改,前提是不改 name(这个参数设定好后不要轻易改),获取的网址也会动态地跟着变,好比改为:

 

1
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 或 models.py 等地方的 reverse函数,一样会根据 name 对应的url获取到新的网址。

想要改网址的时候,修改 urls.py 中的正则表达式部分(url 参数第一部分),name 不变的前提下,其它地方都不须要修改。

 

另外,好比用户收藏夹中收藏的URL是旧的,如何让之前的 /add/3/4/自动跳转到如今新的网址呢?

要知道Django不会帮你作这个,这个须要本身来写一个跳转方法

具体思路是,在 views.py 写一个跳转的函数:

1
2
3
4
5
6
7
8
9
from  django.http  import  HttpResponseRedirect
from  django.core.urlresolvers  import  reverse   # Django 1.4.x - Django 1.10.x
#  from django.urls import reverse  # Django 1.10.x - Django 2.x
 
 
def  old_add2_redirect(request, a, b):
     return  HttpResponseRedirect(
         reverse( 'add2' , args = (a, b))
     )

urls.py中:

1
2
     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/ 了

相关文章
相关标签/搜索