关于Django中路由层(URL-conf)
关于路由层
路由层其实就是项目中urls.py那个文件,那里面存放着url和视图函数的对应关系
它的本质是URL与要为该URL调用的视图函数之间的映射表;你就是以这种方式告诉Django,对于客户端发来的某个URL调用哪一段逻辑代码(视图函数)对应执行。
详解
from django.contrib import admin #文件默认有的 from django.urls import path #默认的一种url控制模式,不能只用正则,若非要使用正则,使用re_path from django.conf.urls import url,include #低版本默认使用的url控制模式,默认可以使用正则,在高版本也可以使用 from app import views #因为url与url要调用的视图函数存在映射关系,而视图函数存在views函数,所以要导入进来 urlpatterns = [ path('admin/', admin.site.urls), # url(r'^articles/2003/$',views.special_case_2003), #正则匹配路径 #无名分组,加括号 # url(r'articles/(\d{4})/$',views.year), #year(request,y) 在view的year函数要记得设置形参来接收 # url(r'articles/(\d{4})/(\d+)/$' ,views.year_month), #year(request,y,m) 在views的函数要设置形参 #有名分组 #形参的设置要和尖括号这边取的名字对应的 # url(r'^articles/(?P<year>\d{4})/(?P<month>\d+)/$',views.year_month2), #year_month2(request,year,month) #可以调换位置,按关键字传参, # 分发 url('app/',include('app.urls')), #当项目太大,所有的url放在这边会不好维护,所以项目足够大就要分发 url('app02/',include('app02.urls')) #然后在各个应用中再新建一个vies文件,有自己的url控制,映射到整个项目urls分发 #每个应用在自己的文件夹创建urls时候,也要导入自己的视图函数 ]
注意:
- 若要从URL 中捕获一个值,只需要在它周围放置一对圆括号。
- 不需要添加一个前导的反斜杠,因为每个URL 都有。例如,应该是
^articles
而不是^/articles
。 - 每个正则表达式前面的'r' 是可选的但是建议加上。它告诉Python 这个字符串是“原始的” —— 字符串中任何字符都不应该转义
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),
]
考虑下这样的两个问题: 第一个问题,函数 year_archive 中year参数是字符串类型的,因此需要先转化为整数类型的变量值,当然year=int(year) 不会有诸如如TypeError或者ValueError的异常。那么有没有一种方法,在url中,使得这一转化步骤可以由Django自动完成? 第二个问题,三个路由中article_id都是同样的正则表达式,但是你需要写三遍,当之后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),
]
基本规则:
- 使用尖括号(
<>
)从url中捕获值。 - 捕获值中可以包含一个转化器类型(converter type),比如使用
<int:name>
捕获一个整数变量。若果没有转化器,将匹配任何字符串,当然也包括了/
字符。 - 无需添加前导斜杠。
关于反向解析
在使用Django 项目时,一个常见的需求是获得URL 的最终形式,以用于嵌入到生成的内容中(视图中和显示给用户的URL等)或者用于处理服务器端的导航(重定向等)。人们强烈希望不要硬编码这些URL(费力、不可扩展且容易产生错误)或者设计一种与URLconf 毫不相关的专门的URL 生成机制,因为这样容易导致一定程度上产生过期的URL。
在需要URL 的地方,对于不同层级,Django 提供不同的工具用于URL 反查:
在模板中:使用 url 模板标签。 {% url '别名' %}
#在template的某一个页面中使用, 先在url分发是给路径起个别名,name=别名, 在页面要使用到这个路径的时候{% url '别名'%} (可以理解成占位符)
#这样无论路径怎么别,我们都是去解析他的别名,,最终找到路径
在Python 代码中:使用from django.urls import reverse()函数
#在脚本中(就是在views视图函数中使用) , 一样的,先给路径起个别名,然后导入reverse函数, 然后用一个变量来接收反向解析的结果 , _url=reverse("别名")
redirect(_url),这样的话,无论路径怎么变化,我们都是通过别名反向解析出他的路径,永远都能找到我们要用的页面
url.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 %}">2012 Archive</a> <ul> {% for yearvar in year_list %} <li><a href="{% url 'news-year-archive' yearvar %}">{{ 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
。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 25岁的心里话
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 闲置电脑爆改个人服务器(超详细) #公网映射 #Vmware虚拟网络编辑器