在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代表所要匹配的模式。例如,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(r'^articles/([0-9]{4})/$', views.year_archive, name='news-year-archive'),
Django在不同的层次也提供了一些工具来实现URL的反向解析。
from django.conf.urls import url from django.contrib import admin import echo.views urlpatterns = [ url(r'^admin/', admin.site.urls), #内容显示,并通过定义name,来进行反向解析 url(r'^lists/(?P<table>\w+)/$', echo.views.lists, name='lists'), #增加内容 url(r'^add/(?P<table>\w+)/$', echo.views.add, name='add'), ]
2) 修改views的函数的参数,在request后加入table,使该函数能够用于所有表格。request是views函数中必须要有的参数。
views.py:
# -*- coding: UTF-8 -*- from .models import Node,Line,Device from forms import NodeForm,LineForm,DeviceForm from django.shortcuts import render, redirect # Create your views here. def lists(request, table): #从根据不同的请求,来获取相应的数据,并跳转至相应页面 if table == 'node': data = Node.objects.all() list_template = 'node_list.html' if table == 'line': data = Line.objects.all() list_template = 'line_list.html' if table == 'device': data = Device.objects.all() list_template = 'device_list.html' #建立context字典,将值传递到相应页面 context = { 'data': data, } #跳转到相应页面,并将值传递过去 return render(request,list_template,context) def add(request, table): #根据提交的请求不同,获取来自不同Form的表单数据 if table == 'node': form = NodeForm(request.POST or None) if table == 'line': form = LineForm(request.POST or None) if table == 'device': form = DeviceForm(request.POST or None) #判断form是否有效 if form.is_valid(): #创建实例,需要做些数据处理,暂不做保存 instance = form.save(commit=False) #将登录用户作为登记人 if table == 'node': instance.node_signer = request.user if table == 'line': instance.line_signer = request.user if table == 'device': instance.device_signer = request.user #保存该实例 instance.save() #跳转至列表页面,配合table参数,进行URL的反向解析 return redirect('lists', table=table) #创建context来集中处理需要传递到页面的数据 context = { 'form': form, } #如果没有有效提交,则仍留在原来页面 return render(request, 'add.html', context)
<pre name="code" class="html"><!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8"> <title></title> </head> <body> <form method='POST' action=''>{% csrf_token %} {{ form }} <input type='submit' value='提交' /> </form> </body> </html>
<pre name="code" class="html"><!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8"> <title></title> </head> <body> <table> <tr> <th>设备名称</th> <th>设备型号</th> </tr> {% for item in data %} <tr> <td>{{ item.device_caption }}</td> <td>{{ item.device_type }}</td> </tr> {% endfor %} </table> </body> </html>
<!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8"> <title></title> </head> <body> <table> <tr> <th>线路名称</th> <th>线路速率</th> <th>线路类型</th> </tr> {% for item in data %} <tr> <td>{{ item.line_code }}</td> <td>{{ item.line_speed }}</td> <td>{{ item.line_type }}</td> </tr> {% endfor %} </table> </body> </html>
<!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8"> <title></title> </head> <body> <table> <tr> <th>节点名称</th> <th>节点地址</th> <th>节点类型</th> </tr> {% for item in data %} <tr> <td>{{ item.node_name }}</td> <td>{{ item.node_address }}</td> <td>{{ item.node_type }}</td> </tr> {% endfor %} </table> </body> </html>