关于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

原文地址:https://www.cnblogs.com/tjp40922/p/10197982.html