六.Django中URL的参数传递

在Django中有非常强大的URL模块,可以按照开发者的想法来制定清晰的URL,同时支持正则表达式。此外,在URL中还可以传递参数。

 

1.      Django处理请求的方式

1) Django通过URLconf模块来进行判断。通常情况下,这就是ROOT_URLCONF配置的价值,但是如果请求携带了一个urlconf的属性(通常被中间件设置),那么这个被携带的urlconf将会替代ROOT_URLCONF的配置。

2) Django会调用Python模块并寻找各种urlpatterns。这是一个属于django.conf.urls.url()实例的python列表。

3) Django会遍历每个URL pattern,自上而下,并且选取收割匹配请求URL的pattern。

4) 一旦匹配某个url pattern的正则表达式,Django将导入并调用相关的view(这是一个简单的python函数,或者是一个class-based view)

这个view将会传递下列参数:

l  一个HttpRequest的实例

l  如果匹配了URL中一个no named group,那么参数将会按根据URL中的位置一一对应

l  如果匹配了URL中一个named group,且参数传递是通过named group来匹配的,那么参数将会被指定的kwargs代替。

5)  如果没有任何一个正则表达式被匹配,那么Django会抛出异常,并报错。

 

2.      URL中的named  group

URL可以通过named group方式传递指定参数,语法为: (?P<name>pattern), name 可以理解为所要传递的参数的名称,pattern代表所要匹配的模式。例如,

[python]  view plain  copy
  1. url(r'^articles/(?P<year>[0-9]{4})/(?P<month>[0-9]{2})/$', views.month_archive),  
那么year,month将会对应views传递过来的year,month的值,而后面紧跟的则代表正则表达匹配的模式。


3. URL的反向解析
通常来说在处理完一个表单之后,网页会发生跳转。通常写URL我们都避免硬编码,这样不方便后期的调整。通常我们需要从URL获取两种内容,最主要是view能够通过URL获取一些标识并处理,另一些信息则是传递过来的参数。
Django提供了一种解决方案,URL mapper是与URL设计一一对应。你可以通过URLconf来实现,并反向使用它。例如,

  • 由用户通过浏览器发起URL请求,调用view,并将URL中的参数传递给view
  • 通过view并附上相应参数,找到相应匹配的URL。
后者我们称之为对URLs的反向解析。反向解析的例子,
[python]  view plain  copy
  1. url(r'^articles/([0-9]{4})/$', views.year_archive, name='news-year-archive'),  

Django在不同的层次也提供了一些工具来实现URL的反向解析。

  • 在template中:使用url标签
  • 在python中:使用django.core.urlresolvers.reverse()函数
  • 在更高层次处理model实例时,可以使用get_absolute_url()方法

4. 利用URL来完成Device,Line表的增加功能
对于信息系统,我们可以把Node,Device, Line都看成是一种资源,对其中任何一种要素的修改,都是对资源的修改,只是会落实到不同的表中,但是在程序中可以一样看到。现在,我们就基于上一节的代码做些修改。


1)修改URL的配置,将原来的add对应的url进行扩充
urls.py:

[python]  view plain  copy
  1. from django.conf.urls import url  
  2. from django.contrib import admin  
  3. import echo.views  
  4.   
  5. urlpatterns = [  
  6.     url(r'^admin/', admin.site.urls),  
  7.     #内容显示,并通过定义name,来进行反向解析  
  8.     url(r'^lists/(?P<table>\w+)/$', echo.views.lists, name='lists'),  
  9.     #增加内容  
  10.     url(r'^add/(?P<table>\w+)/$', echo.views.add, name='add'),  
  11. ]  


2) 修改views的函数的参数,在request后加入table,使该函数能够用于所有表格。request是views函数中必须要有的参数。
views.py:

[python]  view plain  copy
  1. # -*- coding: UTF-8 -*-  
  2. from .models import Node,Line,Device  
  3. from forms import NodeForm,LineForm,DeviceForm  
  4. from django.shortcuts import render, redirect  
  5. # Create your views here.  
  6.   
  7. def lists(request, table):  
  8.     #从根据不同的请求,来获取相应的数据,并跳转至相应页面  
  9.     if table == 'node':  
  10.         data = Node.objects.all()  
  11.         list_template = 'node_list.html'  
  12.     if table == 'line':  
  13.         data = Line.objects.all()  
  14.         list_template = 'line_list.html'  
  15.     if table == 'device':  
  16.         data = Device.objects.all()  
  17.         list_template = 'device_list.html'  
  18.     #建立context字典,将值传递到相应页面  
  19.     context = {  
  20.         'data': data,  
  21.     }  
  22.     #跳转到相应页面,并将值传递过去  
  23.     return render(request,list_template,context)  
  24.   
  25. def add(request, table):  
  26.   
  27.     #根据提交的请求不同,获取来自不同Form的表单数据  
  28.     if table == 'node':  
  29.         form = NodeForm(request.POST or None)  
  30.     if table == 'line':  
  31.         form = LineForm(request.POST or None)  
  32.     if table == 'device':  
  33.         form = DeviceForm(request.POST or None)  
  34.     #判断form是否有效  
  35.     if form.is_valid():  
  36.         #创建实例,需要做些数据处理,暂不做保存  
  37.         instance = form.save(commit=False)  
  38.         #将登录用户作为登记人  
  39.         if table == 'node':  
  40.             instance.node_signer = request.user  
  41.         if table == 'line':  
  42.             instance.line_signer = request.user  
  43.         if table == 'device':  
  44.             instance.device_signer = request.user  
  45.         #保存该实例  
  46.         instance.save()  
  47.         #跳转至列表页面,配合table参数,进行URL的反向解析  
  48.         return redirect('lists', table=table)  
  49.   
  50.     #创建context来集中处理需要传递到页面的数据  
  51.     context = {  
  52.         'form': form,  
  53.     }  
  54.     #如果没有有效提交,则仍留在原来页面  
  55.     return render(request, 'add.html',  context)  




3)在template中建立相关页面:
add.html:

[python]  view plain  copy
  1. <pre name="code" class="html"><!DOCTYPE html>  
  2. <html>  
  3. <head lang="en">  
  4.     <meta charset="UTF-8">  
  5.     <title></title>  
  6. </head>  
  7. <body>  
  8.  <form method='POST' action=''>{% csrf_token %}  
  9.         {{ form }}  
  10.         <input type='submit' value='提交' />  
  11.  </form>  
  12.   
  13. </body>  
  14. </html>  


 
 
device_list.html:

[python]  view plain  copy
  1. <pre name="code" class="html"><!DOCTYPE html>  
  2. <html>  
  3. <head lang="en">  
  4.     <meta charset="UTF-8">  
  5.     <title></title>  
  6. </head>  
  7. <body>  
  8.     <table>  
  9.         <tr>  
  10.             <th>设备名称</th>  
  11.             <th>设备型号</th>  
  12.         </tr>  
  13.         {% for item in data %}  
  14.             <tr>  
  15.                 <td>{{ item.device_caption }}</td>  
  16.                 <td>{{ item.device_type }}</td>  
  17.             </tr>  
  18.         {% endfor %}  
  19.     </table>  
  20. </body>  
  21. </html>  

 
 
line_list.html:

[html]  view plain  copy
  1. <!DOCTYPE html>  
  2. <html>  
  3. <head lang="en">  
  4.     <meta charset="UTF-8">  
  5.     <title></title>  
  6. </head>  
  7. <body>  
  8.     <table>  
  9.         <tr>  
  10.             <th>线路名称</th>  
  11.             <th>线路速率</th>  
  12.             <th>线路类型</th>  
  13.         </tr>  
  14.         {% for item in data %}  
  15.             <tr>  
  16.                 <td>{{ item.line_code }}</td>  
  17.                 <td>{{ item.line_speed }}</td>  
  18.                 <td>{{ item.line_type }}</td>  
  19.             </tr>  
  20.         {% endfor %}  
  21.     </table>  
  22. </body>  
  23. </html>  

node_list.html:

[html]  view plain  copy
  1. <!DOCTYPE html>  
  2. <html>  
  3. <head lang="en">  
  4.     <meta charset="UTF-8">  
  5.     <title></title>  
  6. </head>  
  7. <body>  
  8.     <table>  
  9.         <tr>  
  10.             <th>节点名称</th>  
  11.             <th>节点地址</th>  
  12.             <th>节点类型</th>  
  13.         </tr>  
  14.         {% for item in data %}  
  15.             <tr>  
  16.                 <td>{{ item.node_name }}</td>  
  17.                 <td>{{ item.node_address }}</td>  
  18.                 <td>{{ item.node_type }}</td>  
  19.             </tr>  
  20.         {% endfor %}  
  21.     </table>  
  22. </body>  
  23. </html>  
  24. https://blog.csdn.net/alex_chen_16/article/details/50850435

猜你喜欢

转载自blog.csdn.net/uwenhao2008/article/details/80686149
今日推荐