自定义URL(path)转换器笔记:
需求1
实现一个读取文章列表的demo,用户可以根据/articles/文章分类/
的方式来获取文章.其中文章分类采用的是分类1+分类2+分类3...
的方式拼接的,并且如果只有一个分类,那就不需要加号,示例如下:
# 第一种: 获取python分类下的文章
/articles/python/
# 第二种: 获取python和Django分类下的文章
/articles/python+django/
# 第三种: 获取python和Django和flask分类下的文章
/articles/python+django+flask/
以此类推...
需求2
1.在’文章分类’参数传到视图函数之前要把这些分类分开来存储到列表中.比如参数是python+django
,那么传到视图函数的时候就要变成['python','django']
2.以后在使用reverse
反转的时候,文章分类
的参数不应该是列表了,因为url路径没有用列表
,所有要将视图函数的['python','django']
在转变成python+django
的形式
自定义url转换器
之前已经学到过一些django内置的url转换器,包括有int、uuid等。有时候这些内置的url转换器并不能满足我们的需求,因此django给我们提供了一个接口可以让我们自己定义自己的url转换器。
自定义url转换器按照以下五个步骤来走就可以了:
1.定义一个类,直接继承object就可以了。
2. 在类中定义一个属性regex,这个属性是用来限制url转换器规则的正则表达式。
3. 实现to_python(self,value)
方法,这个方法是将url中的值转换一下,然后传给视图函数的。
4.实现to_url(self,value)方法,这个方法是在做url反转的时候,将传进来的参数转换后拼接成一个正确的url。
5.将定义好的转换器,使用django.urls.converter.register_conterver
方法注册到django中。
示例代码如下:
from django.urls import register_converter
class CategoryConverter(object):
regex = '\w+|(\w+\+\w+)+'
def to_python(self,value):
# 在url中是 python+django+flask
# 转换到视图中需要是["python","django","flask"]
result = value.split("+")
return result
def to_url(self,value):
# 目前是["python","django","flask"]
# 反转url的时候需要是python+django+flask
if isinstance(value,list):
result = "+".join(value)
return result
else:
raise RuntimeError("转换成url的时候,分类参数必须是列表!!!")
register_converter(CategoryConverter,"cate")
URL映射的时候制定默认参数:
使用path
或者是re_path
后,在route中都可以包含参数,而有时候想指定默认的参数,这时候可以通过以下方式来完成。示例代码如下:
from django.urls import path
from . import views
urlpatterns = [
path('blog/', views.page),
path('blog/page<int:num>/', views.page),
]
# View (in blog/views.py)
def page(request, num=1):
# Output the appropriate page of blog entries, according to num.
...
当在访问blog/
的时候,因为没有传递num参数,所以会匹配到第一个url
,这时候就执行view.page
这个视图函数,而在page
函数中,又有num=1
这个默认参数。因此这时候就可以不用传递参数。而如果访问blog/1
的时候,因为在传递参数的时候传递了num
,因此会匹配到第二个url
,这时候也会执行views.page
,然后把传递进来的参数传给page
函数中的num
。