zoukankan      html  css  js  c++  java
  • Django学习笔记之视图高级-类视图

    类视图

    在写视图的时候,Django除了使用函数作为视图,也可以使用类作为视图。使用类视图可以使用类的一些特性,比如继承等。

    View

    django.views.generic.base.View是主要的类视图,所有的类视图都是继承自他。如果我们写自己的类视图,也可以继承自他。然后再根据当前请求的method,来实现不同的方法。比如这个视图只能使用get的方式来请求,那么就可以在这个类中定义get(self,request,*args,**kwargs)方法。以此类推,如果只需要实现post方法,那么就只需要在类中实现post(self,request,*args,**kwargs)。示例代码如下:

    from django.views import View
    class BookDetailView(View):
        def get(self,request,*args,**kwargs):
            return render(request,'detail.html')
    

    类视图写完后,还应该在urls.py中进行映射,映射的时候就需要调用View的类方法as_view()来进行转换。示例代码如下:

    urlpatterns = [        
        path("detail/<book_id>/",views.BookDetailView.as_view(),name='detail')
    ]
    

    除了get方法,View还支持以下方法['get','post','put','patch','delete','head','options','trace']

    如果用户访问了View中没有定义的方法。比如你的类视图只支持get方法,而出现了post方法,那么就会把这个请求转发给http_method_not_allowed(request,*args,**kwargs)。示例代码如下:

    class AddBookView(View):
        def post(self,request,*args,**kwargs):
            return HttpResponse("书籍添加成功!")
    
        def http_method_not_allowed(self, request, *args, **kwargs):
            return HttpResponse("您当前采用的method是:%s,本视图只支持使用post请求!" % request.method)
    

    urls.py中的映射如下:

    path("addbook/",views.AddBookView.as_view(),name='add_book')
    

    如果你在浏览器中访问addbook/,因为浏览器访问采用的是get方法,而addbook只支持post方法,因此以上视图会返回您当前采用的method是:GET,本视图只支持使用post请求!。

    其实不管是get请求还是post请求,都会走dispatch(request,*args,**kwargs)方法,所以如果实现这个方法,将能够对所有请求都处理到。

    TemplateView

    django.views.generic.base.TemplateView,这个类视图是专门用来返回模版的。在这个类中,有两个属性是经常需要用到的,一个是template_name,这个属性是用来存储模版的路径,TemplateView会自动的渲染这个变量指向的模版。另外一个是get_context_data,这个方法是用来返回上下文数据的,也就是在给模版传的参数的。示例代码如下:

    from django.views.generic.base import TemplateView
    
    class HomePageView(TemplateView):
    
        template_name = "home.html"
    
        def get_context_data(self, **kwargs):
            context = super().get_context_data(**kwargs)
            context['username'] = "黄勇"
            return context
    

    urls.py中的映射代码如下:

    from django.urls import path
    
    from myapp.views import HomePageView
    
    urlpatterns = [
        path('', HomePageView.as_view(), name='home'),
    ]
    

    如果在模版中不需要传递任何参数,那么可以直接只在urls.py中使用TemplateView来渲染模版。示例代码如下:

    from django.urls import path
    from django.views.generic import TemplateView
    
    urlpatterns = [
        path('about/', TemplateView.as_view(template_name="about.html")),
    ]
    

    ListView

    在网站开发中,经常会出现需要列出某个表中的一些数据作为列表展示出来。比如文章列表,图书列表等等。在Django中可以使用ListView来帮我们快速实现这种需求。示例代码如下:

    class ArticleListView(ListView):
        model = Article
        template_name = 'article_list.html'
        paginate_by = 10
        context_object_name = 'articles'
        ordering = 'create_time'
        page_kwarg = 'page'
    
        def get_context_data(self, **kwargs):
            context = super(ArticleListView, self).get_context_data(**kwargs)
            print(context)
            return context
    
        def get_queryset(self):
            return Article.objects.filter(id__lte=89)
    

    对以上代码进行解释:

    1. 首先ArticleListView是继承自ListView
    2. model:重写model类属性,指定这个列表是给哪个模型的。
    3. template_name:指定这个列表的模板。
    4. paginate_by:指定这个列表一页中展示多少条数据。
    5. context_object_name:指定这个列表模型在模板中的参数名称。
    6. ordering:指定这个列表的排序方式。
    7. page_kwarg:获取第几页的数据的参数名称。默认是page
    8. get_context_data:获取上下文的数据。
    9. get_queryset:如果你提取数据的时候,并不是要把所有数据都返回,那么你可以重写这个方法。将一些不需要展示的数据给过滤掉。

    给类视图添加装饰器

    在开发中,有时候需要给一些视图添加装饰器。如果用函数视图那么非常简单,只要在函数的上面写上装饰器就可以了。但是如果想要给类添加装饰器,那么可以通过以下两种方式来实现:

    装饰dispatch方法

    from django.utils.decorators import method_decorator
    
    def login_required(func):
        def wrapper(request,*args,**kwargs):
            if request.GET.get("username"):
                return func(request,*args,**kwargs)
            else:
                return redirect(reverse('index'))
        return wrapper
    
    
    class IndexView(View):
        def get(self,request,*args,**kwargs):
            return HttpResponse("index")
    
        @method_decorator(login_required)
        def dispatch(self, request, *args, **kwargs):
            super(IndexView, self).dispatch(request,*args,**kwargs)
    
    

    直接装饰在整个类上

    from django.utils.decorators import method_decorator
    def login_required(func):
        def wrapper(request,*args,**kwargs):
            if request.GET.get("username"):
                return func(request,*args,**kwargs)
            else:
                return redirect(reverse('login'))
        return wrapper
    
    
    @method_decorator(login_required,name='dispatch')
    class IndexView(View):
        def get(self,request,*args,**kwargs):
            return HttpResponse("index")
    
        def dispatch(self, request, *args, **kwargs):
            super(IndexView, self).dispatch(request,*args,**kwargs)
    
    

    分页

    Django中实现分页功能非常简单。因为Django已经内置了两个处理分类的类。分别是PaginatorPagePaginator用来管理整个分类的一些属性,Page用来管理当前这个分页的一些属性。通过这两个类,就可以轻松的实现分页效果。以下对这两个类进行讲解。

    Paginator

    Paginator是用来控制整个分页的逻辑的。比如总共有多少页,页码区间等等。都可以从他上面来获取。

    创建Paginator对象

    class Paginator(object_list, per_page, orphans=0, allow_empty_first_page=True),其中的参数解释如下:

    1. object_list:列表,元组,QuerySet或者是任何可以做切片操作的对象。会将这个里面的对象进行分页。
    2. per_page:分页中,一页展示多少条数据。
    3. orphans:用来控制最后一页元素的个人如果少于orphans指定的个数的时候,就会将多余的添加到上一页中。
    4. allow_empty_first_page:如果object_list没有任何数据,并且这个参数设置为True,那么就会抛出EmptyPage异常。

    常用属性和方法

    1. Paginator.page(number):获取number这页的Page对象。
    2. count:传进来的object_list总共的数量。
    3. num_pages:总共的页数。
    4. page_range:页码的列表。比如[1,2,3,4]

    Page

    常用属性和方法

    1. has_next():是否还有下一页。
    2. has_previous():是否还有上一页。
    3. next_page_number():下一页的页码。
    4. previous_page_number():上一页的页码。
    5. object_list:在当前这页上的对象列表。
    6. number:当前的页码。
    7. paginator:获取Paginator对象。
  • 相关阅读:
    luogu_P1850 换教室
    luogu_P3224 [HNOI2012]永无乡
    luogu_P1064 金明的预算方案
    luogu_P2014 选课
    luogu_P3372 【模板】线段树 1(动态开点)
    luogu_P2852 [USACO06DEC]牛奶模式Milk Patterns
    luogu_P1941 飞扬的小鸟
    luogu_P2678 跳石头
    luogu_P1638 逛画展
    【Tyvj2046】掷骰子
  • 原文地址:https://www.cnblogs.com/biechishaobing/p/10771569.html
Copyright © 2011-2022 走看看