Django之路由层的实现


Posted in Python onSeptember 09, 2019

URL配置(URLconf)就像Django所支撑网站的目录。它的本指是URL与要为该URL调用的视图函数之间的映射表,你就是以这种方式告诉Django,对于客户端发来的某个URL调用哪一段逻辑代码对应执行。

一、简单的路由配置

from django.urls import path,re_path

from app01 import views

urlpatterns = [
re_path(r'^articles/2003/$', views.special_case_2003),
re_path(r'^articles/([0-9]{4})/$', views.year_archive),
re_path(r'^articles/([0-9]{4})/([0-9]{2})/$', views.month_archive),
re_path(r'^articles/([0-9]{4})/([0-9]{2})/([0-9]+)/$', views.article_detail),
]

注:若要从URL中捕获一个值,只需要在它周围放置一对圆括号。不需要添加一个前导的反斜杠,因为每个URL都有。例如,应该是 ^articles 而不是 ^/articles 。每个正则表达式前面的 r 是可选的,但是建议加上,它告诉Python这个字符串是“原始的”——字符串中任何字符都不应该转义。

一些请求的例子:

/articles/2005/03/ 请求将匹配列表中的第三个模式。Django 将调用函数views.month_archive(request, '2005', '03')。
/articles/2005/3/ 不匹配任何URL 模式,因为列表中的第三个模式要求月份应该是两个数字。
/articles/2003/ 将匹配列表中的第一个模式不是第二个,因为模式按顺序匹配,第一个会首先测试是否匹配。请像这样自由插入一些特殊的情况来探测匹配的次序。
/articles/2003 不匹配任何一个模式,因为每个模式要求URL 以一个反斜线结尾。
/articles/2003/03/03/ 将匹配最后一个模式。Django 将调用函数views.article_detail(request, '2003', '03', '03')。

二、有名分组

上面的示例使用简单的、没有命名的正则表达式(通过圆括号)来捕获URL中的额值并以位置参数传递给视图。在更高级的用法中,可以使用命名的正则表达式组来捕获URL中的值并以关键字参数传递给视图。在Python正则表达式中,命名正则表达式组的语法是 (?P<name>pattern) ,其中name是组的名称,pattern是要匹配的模式。下面是以上URLconf使用命名组的重写:

from django.urls import path,re_path

from app01 import views

urlpatterns = [
re_path(r'^articles/2003/$', views.special_case_2003),
re_path(r'^articles/(?P<year>[0-9]{4})/$', views.year_archive),
re_path(r'^articles/(?P<year>[0-9]{4})/(?P<month>[0-9]{2})/$', views.month_archive),
re_path(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')。
/articles/2003/03/03/ 请求将调用函数views.article_detail(request, year='2003', month='03', day='03')。

三、分发

'''
At any point, your urlpatterns can “include” other URLconf modules. This
essentially “roots” a set of URLs below other ones.
'''

from django.urls import path,re_path,include
from app01 import views

urlpatterns = [
re_path(r'^admin/', admin.site.urls),
re_path(r'^blog/', include('blog.urls')),
]

四、反向解析

在使用Django项目时,一个常见的需求是获得URL的最终形式,以用于嵌入到生成的内容中(视图中和显示给用户的URL等)或者用于处理服务器端的导航(重定向等)。人们强烈希望不要硬编码这些URL(费力、不可扩展且容易产生错误)或者设计一种与URLconf毫不相关的专门的URL生成机制,因为这样容易导致一定程度上产生过期的URL。在需要URL的地方,对于不同层级,Django提供不同的额工具用于URL反查:

1、在模板中:使用URL模板标签。

2、在Python代码中:使用 from django.urls import reverse 函数 urls.py:

from django.conf.urls import url

from . import views

urlpatterns = [
#...
re_path(r'^articles/([0-9]{4})/$', views.year_archive, name='news-year-archive'),
#...
]

在模板中:

<a href="{% url 'news-year-archive' 2012 %}" rel="external nofollow" >2012 Archive</a>

<ul>
{% for yearvar in year_list %}
<li><a href="{% url 'news-year-archive' yearvar %}" rel="external nofollow" >{{ yearvar }} Archive</a></li>
{% endfor %}
</ul>

在python中:

from django.urls import reverse
from django.http import HttpResponseRedirect

def redirect_to_year(request):
# ...
year = 2006
# ...
return HttpResponseRedirect(reverse('news-year-archive', args=(year,))) # 同redirect("/path/")

当命名你的URL模式时,请确保使用的名称不会与其它应用中名称冲突。如果你的URL模式叫做comment,而另外一个应用中也有一个同样的名称,当你在模板中使用这个名称的时候不能保证将插入哪个URL。在URL名称中加上一个前缀,比如应用的名称,将减少冲突的可能。我们建议使用myapp-comment 而不是 comment。

五、名称空间

命名空间(Namespace)是标识符的可见范围。一个标识符可在多个命名空间中定义,它在不同命名空间中的含义是互不相干的。这样,在一个新的命名空间中可定义任何标识符,它们不会与任何已有的标识符发生冲突,因为已有的定义都处于其它命名空间中。由于name没有作用域,Django在反解URL时,会在项目全局顺序搜索,当查找到第一个name指定URL时,立即返回。我们在开发项目时,会经常使用name属性反解出URL,当不小心在不同的app的urls中定义相同的name时,可能会导致URL反解错误,为了避免这种事情发生,引入了命名空间。

在项目的urls.py中:

urlpatterns = [
re_path(r'^admin/', admin.site.urls),
re_path(r'^app01/', include(("app01.urls",namespace="app01"))),
re_path(r'^app02/', include(("app02.urls",namespace="app02"))),
]

app01.urls:

urlpatterns = [
re_path(r'^index/', index,name="index"),
]

app02.urls:

urlpatterns = [
re_path(r'^index/', index,name="index"),
]

app01.views:

from django.core.urlresolvers import reverse
def index(request):
  return HttpResponse(reverse("app01:index"))

app02.views:

from django.core.urlresolvers import reverse
def index(request):
  return HttpResponse(reverse("app02:index"))

六、Django2.0版本的path

urlpatterns = [
re_path('articles/(?P<year>[0-9]{4})/', year_archive),
re_path('article/(?P<article_id>[a-zA-Z0-9]+)/detail/', detail_view),
re_path('articles/(?P<article_id>[a-zA-Z0-9]+)/edit/', edit_view),
re_path('articles/(?P<article_id>[a-zA-Z0-9]+)/delete/', delete_view),
]

考虑一下这样的两个问题:

1、函数year_archive中year参数是字符串类型,因此需要先转换为整数类型的变量值,当然year = int(year)不会有诸如TypeError或者VaError的异常。那么有没有一种方法,在url中,使得这一转换步骤可以由Django自动完成?

2、三个路由中article_id规则改变后,需要同时修改三处代码,那么有没有一种方法,只需要修改一处即可?

在Django2.0版本中,可以使用path解决以上的两个问题。

简单示例:

from django.urls import path 
from . import views 
urlpatterns = [ 
  path('articles/2003/', views.special_case_2003), 
  path('articles/<int:year>/', views.year_archive), 
  path('articles/<int:year>/<int:month>/', views.month_archive), 
  path('articles/<int:year>/<int:month>/<slug>/', views.article_detail), 
]

基本规则:

1、使用尖括号 <> 从url中捕获值。

2、捕获值中可以包含一个转换器类型(converter type),比如使用<int : name>捕获一个整数变量。如果没有转换器,将匹配任何字符串,当然也包括了 / 字符。

3、无需添加前导斜杠。

以下是根据https://docs.djangoproject.com/en/2.0/topics/http/urls/#example而整理的示例分析表:

Django之路由层的实现

path转换器:

文档原文是Path converters,暂且翻译为转换器。

Django默认支持以下5个转换器:

1、str——匹配除了路径分隔符 / 之外的非空字符串,这是默认的形式。

2、int——匹配正整数,包含0。

3、slug——匹配字母、数字以及横杠、下划线组成的字符串。

4、uuid——匹配格式化的uuid,如075194d3-6885-417e-a8a8-6c931e272f00。

5、path——匹配任何非空字符串,包含了路径分隔符。

注册自定义转换器:

对于一些复杂或者复用的需要,可以定义自己的转换器。

转换器是一个类或者接口,它的要求有三点:

1、regex类属性,字符串类型。

2、to_python(self, value)方法,value是由类属性regex所匹配到的字符串没返回具体的Python变量值,以供Django传递到对应的视图函数中。

3、to_url(self, value)方法,和to_python相反,value是一个具体的Python变量值,返回其字符串,通常用于url反向引用。

示例:

class FourDigitYearConverter: 
  regex = '[0-9]{4}' 
  def to_python(self, value): 
    return int(value) 
  def to_url(self, value): 
    return '%04d' % value

使用register_converter将其注册到URL配置中:

from django.urls import register_converter, path 
from . import converters, views 
register_converter(converters.FourDigitYearConverter, 'yyyy') 
urlpatterns = [ 
  path('articles/2003/', views.special_case_2003), 
  path('articles/<yyyy:year>/', views.year_archive), 
  ... 
]

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
Python 元类使用说明
Dec 18 Python
python网络编程学习笔记(五):socket的一些补充
Jun 09 Python
python 字典(dict)遍历的四种方法性能测试报告
Jun 25 Python
Flask框架Jinjia模板常用语法总结
Jul 19 Python
pyQT5 实现窗体之间传值的示例
Jun 20 Python
使用pyinstaller打包PyQt4程序遇到的问题及解决方法
Jun 24 Python
Python对列表的操作知识点详解
Aug 20 Python
python 实现多线程下载m3u8格式视频并使用fmmpeg合并
Nov 15 Python
Python之字典对象的几种创建方法
Sep 30 Python
python 根据列表批量下载网易云音乐的免费音乐
Dec 03 Python
详解Python+Selenium+ChromeDriver的配置和问题解决
Jan 19 Python
Python可变集合和不可变集合的构造方法大全
Dec 06 Python
python中web框架的自定义创建
Sep 08 #Python
python web框架中实现原生分页
Sep 08 #Python
python中open函数的基本用法示例
Sep 07 #Python
Python3显示当前时间、计算时间差及时间加减法示例代码
Sep 07 #Python
利用python计算时间差(返回天数)
Sep 07 #Python
Django配置MySQL数据库的完整步骤
Sep 07 #Python
Django项目创建到启动详解(最全最详细)
Sep 07 #Python
You might like
学习php设计模式 php实现策略模式(strategy)
2015/12/07 PHP
简介PHP的Yii框架中缓存的一些高级用法
2016/03/29 PHP
php插入含有特殊符号数据的处理方法
2016/11/24 PHP
php判断IP地址是否在多个IP段内
2020/08/18 PHP
Extjs优化(二)Form表单提交通用实现
2013/04/15 Javascript
js操作label给label赋值及取label的值示例
2013/11/07 Javascript
checkbox勾选判断代码分析
2014/06/11 Javascript
JQuery中serialize()、serializeArray()和param()方法示例介绍
2014/07/31 Javascript
jQuery遍历之next()、nextAll()方法使用实例
2014/11/08 Javascript
jQuery中:not选择器用法实例
2014/12/30 Javascript
js实现按钮控制图片360度翻转特效的方法
2015/02/17 Javascript
jquery实现漂亮的二级下拉菜单代码
2015/08/26 Javascript
学习Javascript面向对象编程之封装
2016/02/23 Javascript
JS实现兼容各种浏览器的获取选择文本的方法【测试可用】
2016/06/21 Javascript
node+experss实现爬取电影天堂爬虫
2016/11/20 Javascript
jQuery实现6位数字密码输入框
2016/12/29 Javascript
jQuery实现导航栏头部菜单项点击后变换颜色的方法
2017/07/19 jQuery
vue-router实现tab标签页(单页面)详解
2017/10/17 Javascript
webpack4与babel配合使es6代码可运行于低版本浏览器的方法
2018/10/12 Javascript
基于Webpack4和React hooks搭建项目的方法
2019/02/05 Javascript
了解JavaScript表单操作和表单域
2019/05/27 Javascript
Python中的random()方法的使用介绍
2015/05/15 Python
Python多线程扫描端口代码示例
2018/02/09 Python
python使用turtle库与random库绘制雪花
2018/06/22 Python
Python 在 VSCode 中使用 IPython Kernel 的方法详解
2020/09/05 Python
土耳其家居建材网站:Koçtaş
2016/11/22 全球购物
美国环保婴儿用品公司:The Honest Company
2017/11/23 全球购物
Crocs欧洲官网:Crocs Europe
2020/01/14 全球购物
介绍一下Make? 为什么使用make
2013/12/08 面试题
大学生标准推荐信范文
2013/11/25 职场文书
优秀员工获奖感言
2014/03/01 职场文书
技术支持岗位职责
2015/02/13 职场文书
旷工检讨书大全
2015/08/15 职场文书
学校2016年九九重阳节活动总结
2016/04/01 职场文书
python 办公自动化——基于pyqt5和openpyxl统计符合要求的名单
2021/05/25 Python
pytorch model.cuda()花费时间很长的解决
2021/06/01 Python