Django之路由层与视图层

Django之路由层与视图层

路由层

urlpatterns = [
    url(r'^admin/',admin.site.urls),
    url(r'^$',views.home),  # 网页首页路由
    url(r'^test/$',views.test),
    url(r'^testadd/$',views.testadd),
    url(r'',views.error)  # 网页不存在页面
]

# 注意:第一个参数是正则表达式,匹配规则按照从上往下依次匹配,匹配到一个以后当即中止,直接执行对应的视图函数

无名分组

无名分组(将加括号的正则表达式匹配到的内容看成位置参数自动传递给对应的视图函数)html

url(r'^test/(\d+)/',views.test),  # 匹配一个或多个数字

def test(request,xxx):  # xxx接收正则匹配到的数字
    print(xxx)  # 变量名本身定义
    return HttpResponse('ok')

有名分组

有名分组(将加括号的正则表达式匹配到的内容看成关键字参数自动传递给对应的视图函数)前端

url(r'^test/(?P<year>\d+)',views.test)  # 匹配一个或多个数字

def test(request,year):
    print(year)  # 这里的变量根据路由指定的
    return HttpResponse('ok')

注意:无名分组和有名分组不能混着用,可是支持同一类型多个形式匹配python

#无名分组多个
url(r'^test/(\d+)/(\d+)',views.test),

#有名分组多个
url(r'^test/(?P<year>\d+)/(?P<yyy>\d+)/',views.test)

反向解析

反向解析:根据名字动态获取到对应路径,视图层须要导入模块reversegit

#能够给每个路由与视图函数对应关系起一个名字
#这个名字可以惟一标识出对应的路径
#注意名字不能重复,是惟一的
url(r'^index666/$',views.index,name='index')

from django.shortcuts import reverse
#后端使用
reversr('index')
#也能够
reverse('路由与视图函数对应关系起的别名')

#前端使用
{% url 'index' %}
#也能够
{% url '别名' %}

无名分组反向解析

url(r'^test/(\d+)/',views.test,name='list')

#后端使用
reverse('list',args=(10,))  #这边数字能够自定义,能够填数据的主键字段值,方便以后取出编辑

#前段使用
{% url 'list' 10 %}

编辑简单示例

url(r'^edit/(\d+)/',views.edit,name='edit')

# 前端
{%for user_obj in user_list%}
    <a href='edit/{{ user_obj.pk }}/'></a>
{% endfor %}

#后端
from diango.shortcuts import reverse
def edit(request,edit_id):
    url = reverse('edit',args=(edit_id,))
    
#模板
{% url 'edit' edit_id %}

有名分组反向解析

#后端使用
reverse('list',arge=(10,))
#也能够
reverse('list',kwargs={'year':10})  # 了解

#前端使用
{% url 'list' 10 %}
#也能够
{% url 'list' year=10 %}  # 了解

总结:针对有名函数和无名函数的反向解析统一采用一种格式便可正则表达式

#后端
reverse('list',args=(10,))  #这里的数字一般都是数据的主键值

#前端
{% url 'list' 10 %}

反向解析的本质:就是获取到一个可以访问名字所对应的视图函数django

路由分发

简介:django每个app下均可以有本身的urls.py路由层,templates文件夹,static文件夹,项目名下的urls.py(总路由)再也不作路由与视图函数的匹配关系而是作路由的分发json

#在路由层导入模块
from django.conf.urls import url,include
from app01 import urls as app01_urls
from app02 import urls as app02_urls

#路由分发,注意路由分发总路由不能以$结尾
url(r'^app01/',include(app01_urls)),  # 防止重名
url(r'^app02/',include(app02_urls))

#也能够不用起别名
url(r'^app01/',include('app01.urls'))
url(r'^app02/',include('app02.urls'))

#在应用下新建urls.py(约定俗成)文件,在该文件内写路由与视图的对应关系便可
from django.conf.urls  import url
from app01 import views
urlpatterns = [
    url(r'^index/',views.index)
]

名称空间(了解)后端

#总路由
url(r'^app01/',include(app01_urls,namespace='app01')),
url(r'^app02/',include(app02_urls,namespace='app02'))

#app01中urls.py
from django.conf.urls import url
from app01 import views
urlpatterns = [
    url(r'^index/',views.index,name='index')
]

#app02中urls.py
from django.conf.urls import url
from app02 import views
urlpatterns = [
    url(r'^index/',views.index,name='index')
]

#app01中views.py
reverse('app01:index')

#app02中views.py
reverse('app02:index')

#注意:这种了解一下就能够了,不一样app中起不一样的名字就能够了

伪静态网页

搜索优化seoapp

url(r'^index.html',views.index,name='app01_index')
#人为加上后缀,让百度误觉得是静态网页保存起来,增长搜索几率

虚拟环境

不一样的项目配置不一样的python解释器函数

建立虚拟环境:

django1.0与django2.0之间的区别:

  • 1.django2.0里面的path第一个参数不支持正则,写什么就匹配什么,100%精准匹配
  • 2.django2.0里面的re_path对应着django1.0里面的url

django2.0的特色

django2.0里面的path不支持正则表达式,可是它提供了五个默认的转换器:

  • 1.str,匹配除了路径分隔符(/)以外的非空字符串,这是默认形式
  • 2.int,匹配正整数,包含0
  • 3.slug,匹配字母、数字、以及横杠、下划线组成的字符串
  • 4.uuid,匹配格式化的uuid
  • 5.path,匹配任何非空字符串,包含了路径分隔符(/),不能用(?)

自定义转换器

  • 1.正则表达式
  • 2.类
  • 3.注册
from django.contrib import admin
from django.urls import path,re_path,register_converter

class FourDigitYearConverter:
    regex = '[0-9]{4}'
    def to_python(self,value):
        return int(value)
    def to_url(self,value):
        return '%04d'%value  #占4位,不够用0填满,超了则按照原来的位数
   
register_converter(FourDigitYearConverter,'yyyy')


urlpatterns = [
    path('admin/',views.site.urls),
    path('index/<yyyy:id>',views.index)
]

#路由匹配到的数据默认都是字符串形式

视图层

FBV与CBV

FBV:基于函数的视图

CBV:基于类的函数

#CBV:
#路由层
urlpatterns = [
    #url(r'^mycls/',views.view),看下图
    url(r'^mycls/',views.MyCls.as_view())
]

#视图层
class MyCls(View):
    def get(self,request):
        return render(request,'index.html')
    def post(self,request):
        return HttpResponse('ok')

JsonResponse

JsonResponse是HttpResponse的子类,专门用来生成JSON编码的响应

from django.http import JsonResponse

def index(request):
    return JsonResponse({'name':'lucas大帅比','password':'123'},json_dumps_params={'ensure_ascii':False})  # 由于有中文,ensure_ascii保证以原样返回

#json中
print(json.dumps({'name':'lucas大帅比','password':'123'},ensure_ascii=False))

简单的文件上传

前端form表单中须要注意的点:

  • 1.method须要指定成post
  • 2.enctype须要改成formdata格式,enctype='multipart/form-data'

后端暂时须要注意的是:

  • 1.配置文件中注释掉csrfmiddleware中间件
  • 2.经过request.FILES获取用户上传的post文件数据
def upload_file(request):
    if request.method == 'POST':
        file_obj = request.FILES.get('my_file')
        print(file_obj.name)  #获取文件名
        with open(file_obj.name,'wb')as f:
            for line in file_obj.chunks():
            #for line in file_obj:
                f.write(line)
        return HttpResponse('ok')
    return render(request,'index.html')
print(request.path)  #/upload_file/
print(request.get_full_path())  #/upload_file/?name=lucas,获取完整路径
相关文章
相关标签/搜索