Django搭建我的博客:基于类的视图

说是完结,立刻又开始写进阶篇了。html

本章不会为博客项目增长新功能,可是也一样重要,由于咱们要学习高逼格的基于的视图。python

什么是类视图

前面章节中写的全部视图都是基于函数的,即def;而类视图是基于类的,即classgit

有编程基础的同窗都知道,是面向对象技术中很是重要的概念。具备复杂数据功能的类,能够经过继承垂手可得的将自身特性传递给另外一个类,从而实现代码的高效复用。github

相比之前的函数视图,类视图有如下优点:web

  • HTTP方法(GETPOST等)相关的代码,能够经过方法而不是条件分支来组织
  • 能够经过诸如mixins(多重继承)之类的面向对象技术将代码分解为可重用组件

说的都是什么意思?经过例子来感觉一下。数据库

列表

函数和类

假设咱们有一个博客列表,列表既有GET方法、又有POST方法,那么用视图函数看起来像这样:django

views.py

def article_list_example(request):
    """处理GET请求"""
    if request.method == 'GET':
        articles = ArticlePost.objects.all()
        context = {'articles': articles}
        return render(request, 'article/list.html', context)
复制代码

而在类视图中,则变为这样:编程

views.py

from django.views import View

class ArticleListView(View):
    """处理GET请求"""
    def get(self, request):
        articles = ArticlePost.objects.all()
        context = {'articles': articles}
        return render(request, 'article/list.html', context)
复制代码

从本质上讲,基于类的视图容许你使用不一样的类实例方法(即上面的def get())响应不一样的HTTP请求方法,而不须要使用条件分支代码。这样作的好处是把不一样的HTTP请求都分离到独立的函数中,逻辑更加清晰,而且方便复用。ide

须要注意的是,由于Django的URL解析器但愿将请求发送到函数而不是类,因此类视图有一个 as_view()方法,该方法返回一个函数,当请求匹配关联模式的URL时,则调用该函数。函数

即,视图函数的url本来写为:

urls.py

...
urlpatterns = [
    path('...', views.article_list_example, name='...'),
]
复制代码

类视图的url需改写为:

urls.py

...
urlpatterns = [
    path('...', views.ArticleListView.as_view(), name='...'),
]
复制代码

通用视图

列表这样的功能在web开发中是很常见的,开发者会一遍又一遍写几乎相同的列表逻辑。Django的通用视图正是为缓解这种痛苦而开发的。它们对经常使用模式进行抽象,以便你快速编写公共视图,而无需编写太多代码。

所以用列表通用视图改写以下:

views.py

from django.views.generic import ListView

class ArticleListView(ListView):
    # 上下文的名称
    context_object_name = 'articles'
    # 查询集
    queryset = ArticlePost.objects.all()
    # 模板位置
    template_name = 'article/list.html'
复制代码

列表继承了父类ListView,也就得到了父类中的处理列表的方法,所以你能够看到,咱们在本身的类中没有写任何处理的逻辑,仅仅是赋值了几个变量而已。

动态过滤

从数据库中筛选特定的内容也是常见的需求,类视图如何实现呢?

你可能想到了,将上面代码中改成queryset = ArticlePost.objects.filter()就能够了。

除此以外,更好的办法是覆写get_queryset()方法:

views.py

...

class ArticleListView(ListView):
    context_object_name = 'articles'
    template_name = 'article/list.html'

    def get_queryset(self):
        """ 查询集 """
        queryset = ArticlePost.objects.filter(title='Python')
        return queryset
复制代码

例子中只是过滤出标题为“Python”的文章而已,有些大材小用了;可是你能够在get_queryset()中写复杂的联合查询逻辑,知足个性化的功能。

添加上下文

在博客列表的设计时,咱们返回给模板的上下文除了articles之外,还有不少额外的信息,如ordersearch;在类视图中一样能够实现,改写get_context_data()方法便可:

views.py

...

class ArticleListView(ListView):
    ...

    def get_context_data(self, **kwargs):
        # 获取原有的上下文
        context = super().get_context_data(**kwargs)
        # 增长新上下文
        context['order'] = 'total_views'
        return context
复制代码

除此以外,ListView还有些别的方法能够覆写,深刻了解能够看这里:官方文档

混入类

混入类(Mixin)是指具备某些功能、一般不独立使用、提供给其余类继承功能的类。嗯,就是“混入”的字面意思。

前面的列表视图中已经有get_context_data()方法了。假设须要写一个功能相似的视频列表,就能够用Mixin来避免重复代码:

views.py

...

class ContextMixin:
    def get_context_data(self, **kwargs):
        context = super().get_context_data(**kwargs)
        context['order'] = 'total_views'
        return context

class ArticleListView(ContextMixin, ListView):
    ...

class VideoListView(ContextMixin, ListView):
    ...
复制代码

经过混入,两个子类都得到了get_context_data()方法。

从语法上看,混入是经过多重继承实现的。有区别的是,Mixin是做为功能添加到子类中的,而不是做为父类。

实际上Django内置了不少通用的Mixin类,实现了大部分经常使用的功能,点这里深刻了解:官方文档

详情页

既然列表都有通用视图,详情页固然也有对应的DetailView

用类视图写一个简单的详情页

views.py

from django.views.generic import DetailView

class ArticleDetailView(DetailView):
    queryset = ArticlePost.objects.all()
    context_object_name = 'article'
    template_name = 'article/detail.html'
复制代码

而后配置url:

urls.py

...
urlpatterns = [
    # 详情类视图
    path('detail-view/<int:pk>/', views.ArticleDetailView.as_view(), name='...'),
]
复制代码

注意这里传入的参数不是id而是pk,这是视图的要求(也能够传入slug)。pk是数据表的主键,在默认状况下其实就是id

这就写好了!

也能够添加任何别的功能,好比统计浏览量

views.py

...
class ArticleDetailView(DetailView):
    ...
    def get_object(self):
        """ 获取须要展现的对象 """
        # 首先调用父类的方法
        obj = super(ArticleDetailView, self).get_object()
        # 浏览量 +1
        obj.total_views += 1
        obj.save(update_fields=['total_views'])
        return obj
复制代码

方法get_object()的做用是获取须要展现的对象。首先调用父类方法,将这个对象赋值给obj变量,而后再对其进行统计浏览量的操做,最后将对象返回。至关于在原有的方法中把本身的逻辑“塞”了进去。

关于DetailView更多特性看这里:官方文档

编辑

除了可以展现信息,通用视图还包含CreateViewUpdateViewDeleteView编辑数据的类。

若是要新建文章,则视图能够这么写:

views.py

from django.views.generic.edit import CreateView

class ArticleCreateView(CreateView):
    model = ArticlePost
    
    fields = '__all__'
    # 或者只填写部分字段,好比:
    # fields = ['title', 'content']
    
    template_name = 'article/create_by_class_view.html'
复制代码

建立create_by_class_view.html文件(目录在哪,你应该已经很清楚了),写入:

create_by_class_view.html

<form method="post">{% csrf_token %}
    {{ form.as_p }}
    <input type="submit" value="Save">
</form>
复制代码

最后添加url:

urls.py

urlpatterns = [
    path('create-view/', views.ArticleCreateView.as_view(), name='...'),
]
复制代码

虽然外观简陋(这不是重点),但如今这个视图确实已经可以建立新文章了!

UpdateViewDeleteView这里就再也不赘述了,之后用到的地方再进行讲解。

想提早了解的同窗戳这里:官方文档

总结

有没有感觉到代码隔离继承的强大?没有?之后的章节会逐渐使用编写视图,你会慢慢体会的。

类视图的内容很是丰富,短短一篇文章只能走马观花而已。读者在编程中遇到困难了,官方文档是你最好的教程。

若是你有耐心从头至尾阅读类视图的官方文档,那固然是最好的了。


转载请告知做者并注明出处。

相关文章
相关标签/搜索