【Django-URL name详解005】

时间:2022-04-19 01:57:25

1、打开zqxt_views/urls.py

from django.contrib import admin
from django.urls import path
from calc import views as calc_views

urlpatterns = [
    path('add/',calc_views.add,name='add'),
    path('add/<int:a>/<int:b>/',calc_views.add2,name='add2'),
    path('admin/', admin.site.urls),
]
path('add/',calc_views.add,name='add'),这里的name='add'是用来干什么的呢?
简单说,name可以用于templates,models,views....中等到对应的网址,相当于给网址取个名字,只要名字不变,网址变了也能通过名字获取到

为了进一步弄清这个问题,我们先建一个首页的视图和url

2、修改calc/views.py
# Create your views here.
from django.shortcuts import render
from django.http import HttpResponse

#新添加的代码
def index(request):
        return render(request,'home.html')

def add(request):
        a = request.GET["a"]
        b = request.GET["b"]
        c = int(a)+int(b)
        return HttpResponse(str(c))

def add2(request,a,b):
        c = int(a)+int(b)
        return HttpResponse(str(c))
render是渲染模板

2、将'calc'这个app加入到zqxt_views/setting.py中
INSTALLED_APPS = [
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',

    'calc',
]
这样,使用render的时候,Django会自动找到INSTEALLED_APP中列出的各个app下的templates中的文件

小提示,DEBUG=True的时候,Django还可以自动找到各app下的static文件夹中的静态文件(js,css,图片等资源),方便开发


4、我们在calc这个app中新建一个templates文件夹,在templates中新建一个home.html
<!DOCTYPE html>
<html>
<head>
    <title>自强学堂</title>
</head>
<body>

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

</body>
</html>
 

修改zqxt_views/urls.py

urlpatterns = [
    path('index/',calc_views.index,name='home'), #新增加
    path('add/',calc_views.add,name='add'),
    path('add/<int:a>/<int:b>/',calc_views.add2,name='add2'),
    path('admin/', admin.site.urls),
]

python manage.py runserver运行后,访问http://127.0.0.1:8000/网址,可以看到

【Django-URL name详解005】

我们计算加法的时候用的是/add/4/5,后来需求发生变化,比如改成/4_add_5/,但在网页中国,代码中很多地方都写死的/add/4/5,比如模板中可能是这样写的

 

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

 

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

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

我们先说一下如何用Python代码获取对应的网址(可以用在views.py,models.py等各种需要转换得到网址的地方):

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

python manage.py shell
>>> from django.urls import reverse
>>> reverse('add2',args=(4,5))
'/add/4/5/'
>>> reverse('add2',args=(555,666))
'/add/555/666/'
>>> 

reverse接受url中的name作为第一个参数,我们在代码中就可以通过reverse()来获取对应的网址(这个网址可以用来跳转,也可以用来计算相关页面的地址),只要对应的url的name不改,就不用改代码中的网址。

在网页模板中也是一样,可以很方便的使用。

不带参数的:
{% url 'name' %}
带参数的:参数可以是变量名
{% url 'name' 参数 %}

例如:
<a href={% url 'add2' 4 5 %}>link</a>

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

<a href="{% url 'add2' 4 5 %}">link<\a>

这样就可以通过{% url 'add2' 4 5 %} 获取到对应的网址/add/4/5/

 

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

#django2.0以上版本不支持正则表达式,必须导入re_path包,否则将不会生效
from django.urls import re_path
re_path('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写一个跳转的函数:

from django.shortcuts import render
from django.http import HttpResponse
from django.urls import reverse  #新导入的包

def index(request):
        return render(request,'home.html')

def add(request):
        a = request.GET["a"]
        b = request.GET["b"]
        c = int(a)+int(b)
        return HttpResponse(str(c))

def add2(request,a,b):
        c = int(a)+int(b)
        return HttpResponse(str(c))

def old_add2_redirect(request,a,b):
        return HttpResponseRedirect(
                reverse('add2',args=(a,b))
        )  #新添加的代码

urls.py中:

re_path('add/(\d+)/(\d+)/',calc_views.old_add2_redirect),
re_path('new_add/(\d+)/(\d+)/',calc_views.add2,name='add2'),

这样,加入用户收藏夹中有/add/4/5/,访问时就会自动跳转到新的/new_add/4/5了

 

开始可能觉的直接写网址简单,但是用多了你一定会发现,用“死网址”的方法很糟糕