1.1 视图混合介绍
1、Mixin和View的职能区分
1. Mixin提供数据,View提供模板和渲染,所以一般get_context_data在Mixin中,get(),post(),head()在View中
2. Mixin和View不是能随意组合的,必须要注意他们之间的方法的解析顺序,也就是MRO
2、ContextMixin
1. 直接就是一个get_context_data,用于返回context数据。
3、View
1. 会调用所有的get方法,post方法,具体是这些['get', 'post', 'put', 'delete', 'head', 'options', 'trace']
2. View中是没有返回一个response的,所以光继承View的话,必须要重写get等,以返回一个response。
4、TemplateResponseMixin
1. 故名思议,这个Mixin会加入Template的基本信息,也就是template的名字
2. 但是光有Template信息是没有用的,因为她没有跟View想联系起来,如果想要跟View联系起来的话必须想
办法把render_to_response插进MRO的调用顺序,而且TemplateResponseMixin是没有context的信息的
3. 有一个可以借鉴的方法就是TemplateView的做法, Mixin和View把原来的试图函数中的三个东西分开了
模板(TemplateResponseMixin),
上下文数据(ContextMixin),
负责将这些联系起来的一个东西(View)
5、TemplateView
1. TemplateView就继承自TemplateResponseMixin,ContextMixin以及View,所以它的调用思路就很明确了
2. 在其中定义一个get方法,然后通过get方法去将上面的三个东西联系在一起.
3. 我们可以看一下TemplateView的源代码
1.2 TemplateView 基本使用
from django.conf.urls import url from django.contrib import admin from app01 import views urlpatterns = [ url(r'^admin/', admin.site.urls), url(r'^home/', views.HomePageView.as_view(),name='home'), ]
from django.db import models class Article(models.Model): name = models.CharField(max_length=32) age = models.SmallIntegerField() def __str__(self): return self.name
from django.views.generic.base import TemplateView from app01.models import Article class HomePageView(TemplateView): template_name = "home.html" #视图返回的页面 def get_context_data(self, **kwargs): #向上下文变量中添加额外的数据 context = super(HomePageView, self).get_context_data(**kwargs) context['latest_articles'] = Article.objects.all()[:5] print(context) return context # 返回给前端页面字典,context内容 context = {'latest_articles':"<QuerySet [<Article: zhangsan>, <Article: lisi>, <Article: wangwu>]>", # 表中获取的数据 'view':" <app01.views.HomePageView object at 0x06A2B470>"} # 视图返回的页面
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <h1>home</h1> {% for article in latest_articles%} {{article.name}} {% endfor %} </body> </html>
1.3 ListView
参考官网: https://docs.djangoproject.com/en/2.0/ref/class-based-views/generic-display/#listview
参考博客: https://www.zmrenwu.com/post/33/
1、ListView作用
作用:ListView 作用是按照指定方式向数据库获取表中数据
2、ListView使用举例
from django.conf.urls import url from django.contrib import admin from app01 import views urlpatterns = [ url(r'^admin/', admin.site.urls), url(r'^home/', views.HomePageView.as_view(),name='home'), ]
# -*- coding: utf-8 -*- from __future__ import unicode_literals from django.db import models class Article(models.Model): name = models.CharField(max_length=32) age = models.SmallIntegerField() def __str__(self): return self.name
# -*- coding: utf-8 -*- from __future__ import unicode_literals from django.shortcuts import render from django.views.generic import ListView from app01.models import Article class HomePageView(ListView): template_name = 'home.html' model = Article context_object_name = 'article_list' # 指定获取的模型列表数据保存的变量名 def get_queryset(self, ): '''get_queryset函数常处理对数据库表查询结果''' print self.request.GET # GET请求的所有参数都可以在这里获取 queryset = self.model.objects.all()[:2] # 根据GET请求的条件对Article表进行过滤 return queryset def get_context_data(self, **kwargs): '''get_context_data 可以携带其他信息,就如常规中设置的字典''' print self.request.GET # GET请求的所有参数都可以在这里获取 context = super(HomePageView, self).get_context_data(**kwargs) context['k1'] = 'v1' context['k2'] = 'v2' return context
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <h1>home</h1> <!--1、后端如果设置:context_object_name = 'article_list' 这里就可以获取到值了 --> <div> {% for article in article_list%} {{article.name}} {% endfor %} </div> <!--2、如果后端没有设置 context_object_name 默认表查询的queryset=object_list --> <div> {% for object in object_list %} {{object.name}} {% endfor %} </div> <!--3、后端通过context传的字典直接可以获取 --> <p>{{ k1 }}</p> <p>{{ k2 }}</p> </body> </html>
1.4 DetailView
1、DetailView作用
1) 除了从数据库中获取模型列表的数据外,从数据库获取模型的一条记录数据也是常见的需求。
2) 比如查看某篇文章的详情,就是从数据库中获取这篇文章的记录然后渲染模板,DetailView
专门提供了这种功能
blog/views.py from django.views.generic import ListView, DetailView # 记得在顶部导入 DetailView class PostDetailView(DetailView): # 这些属性的含义和 ListView 是一样的 model = Post template_name = 'blog/detail.html' context_object_name = 'post' def get(self, request, *args, **kwargs): # 覆写 get 方法的目的是因为每当文章被访问一次,就得将文章阅读量 +1 # get 方法返回的是一个 HttpResponse 实例 # 之所以需要先调用父类的 get 方法,是因为只有当 get 方法被调用后, # 才有 self.object 属性,其值为 Post 模型实例,即被访问的文章 post response = super(PostDetailView, self).get(request, *args, **kwargs) # 将文章阅读量 +1 # 注意 self.object 的值就是被访问的文章 post self.object.increase_views() # 视图必须返回一个 HttpResponse 对象 return response def get_object(self, queryset=None): # 覆写 get_object 方法的目的是因为需要对 post 的 body 值进行渲染 post = super(PostDetailView, self).get_object(queryset=None) post.body = markdown.markdown(post.body, extensions=[ 'markdown.extensions.extra', 'markdown.extensions.codehilite', 'markdown.extensions.toc', ]) return post def get_context_data(self, **kwargs): # 覆写 get_context_data 的目的是因为除了将 post 传递给模板外(DetailView 已经帮我们完成), # 还要把评论表单、post 下的评论列表传递给模板。 context = super(PostDetailView, self).get_context_data(**kwargs) form = CommentForm() comment_list = self.object.comment_set.all() context.update({ 'form': form, 'comment_list': comment_list }) return context