Django搭建個(gè)人博客:基于類的視圖

說是完結(jié),馬上又開始寫進(jìn)階篇了。

本章不會(huì)為博客項(xiàng)目增加新功能,但是也同樣重要,因?yàn)槲覀円獙W(xué)習(xí)高逼格的基于的視圖。

什么是類視圖

前面章節(jié)中寫的所有視圖都是基于函數(shù)的,即def;而類視圖是基于類的,即class。

有編程基礎(chǔ)的同學(xué)都知道,是面向?qū)ο蠹夹g(shù)中非常重要的概念。具有復(fù)雜數(shù)據(jù)、功能的類,可以通過繼承輕而易舉的將自身特性傳遞給另一個(gè)類,從而實(shí)現(xiàn)代碼的高效復(fù)用。

相比以前的函數(shù)視圖,類視圖有以下優(yōu)勢(shì):

  • HTTP方法(GET,POST等)相關(guān)的代碼,可以通過方法而不是條件分支來組織
  • 可以通過諸如mixins(多重繼承)之類的面向?qū)ο蠹夹g(shù)將代碼分解為可重用組件

說的都是什么意思?通過例子來感受一下。

列表

函數(shù)和類

假設(shè)我們有一個(gè)博客列表,列表既有GET方法、又有POST方法,那么用視圖函數(shù)看起來像這樣:

views.py

def article_list_example(request):
    """處理GET請(qǐng)求"""
    if request.method == 'GET':
        articles = ArticlePost.objects.all()
        context = {'articles': articles}
        return render(request, 'article/list.html', context)

而在類視圖中,則變?yōu)檫@樣:

views.py

from django.views import View

class ArticleListView(View):
    """處理GET請(qǐng)求"""
    def get(self, request):
        articles = ArticlePost.objects.all()
        context = {'articles': articles}
        return render(request, 'article/list.html', context)

從本質(zhì)上講,基于類的視圖允許你使用不同的類實(shí)例方法(即上面的def get())響應(yīng)不同的HTTP請(qǐng)求方法,而不需要使用條件分支代碼。這樣做的好處是把不同的HTTP請(qǐng)求都分離到獨(dú)立的函數(shù)中,邏輯更加清晰,并且方便復(fù)用。

需要注意的是,因?yàn)镈jango的URL解析器希望將請(qǐng)求發(fā)送到函數(shù)而不是類,所以類視圖有一個(gè) as_view()方法,該方法返回一個(gè)函數(shù),當(dāng)請(qǐng)求匹配關(guān)聯(lián)模式的URL時(shí),則調(diào)用該函數(shù)。

即,視圖函數(shù)的url原本寫為:

urls.py

...
urlpatterns = [
    path('...', views.article_list_example, name='...'),
]

類視圖的url需改寫為:

urls.py

...
urlpatterns = [
    path('...', views.ArticleListView.as_view(), name='...'),
]

通用視圖

列表這樣的功能在web開發(fā)中是很常見的,開發(fā)者會(huì)一遍又一遍寫幾乎相同的列表邏輯。Django的通用視圖正是為緩解這種痛苦而開發(fā)的。它們對(duì)常用模式進(jìn)行抽象,以便你快速編寫公共視圖,而無需編寫太多代碼。

因此用列表通用視圖改寫如下:

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,也就獲得了父類中的處理列表的方法,因此你可以看到,我們?cè)谧约旱念愔袥]有寫任何處理的邏輯,僅僅是賦值了幾個(gè)變量而已。

動(dòng)態(tài)過濾

從數(shù)據(jù)庫(kù)中篩選特定的內(nèi)容也是常見的需求,類視圖如何實(shí)現(xiàn)呢?

你可能想到了,將上面代碼中改為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

例子中只是過濾出標(biāo)題為“Python”的文章而已,有些大材小用了;但是你可以在get_queryset()中寫復(fù)雜的聯(lián)合查詢邏輯,滿足個(gè)性化的功能。

添加上下文

在博客列表的設(shè)計(jì)時(shí),我們返回給模板的上下文除了articles以外,還有很多額外的信息,如order、search;在類視圖中同樣可以實(shí)現(xiàn),改寫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)是指具有某些功能、通常不獨(dú)立使用、提供給其他類繼承功能的類。嗯,就是“混入”的字面意思。

前面的列表視圖中已經(jīng)有get_context_data()方法了。假設(shè)需要寫一個(gè)功能類似的視頻列表,就可以用Mixin來避免重復(fù)代碼:

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):
    ...

通過混入,兩個(gè)子類都獲得了get_context_data()方法。

從語法上看,混入是通過多重繼承實(shí)現(xiàn)的。有區(qū)別的是,Mixin是作為功能添加到子類中的,而不是作為父類。

實(shí)際上Django內(nèi)置了很多通用的Mixin類,實(shí)現(xiàn)了大部分常用的功能,點(diǎn)這里深入了解:官方文檔

詳情頁

既然列表都有通用視圖,詳情頁當(dāng)然也有對(duì)應(yīng)的DetailView。

用類視圖寫一個(gè)簡(jiǎn)單的詳情頁

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='...'),
]

注意這里傳入的參數(shù)不是id而是pk,這是視圖的要求(也可以傳入slug)。pk是數(shù)據(jù)表的主鍵,在默認(rèn)情況下其實(shí)就是id

這就寫好了!

也可以添加任何別的功能,比如統(tǒng)計(jì)瀏覽量

views.py

...
class ArticleDetailView(DetailView):
    ...
    def get_object(self):
        """
        獲取需要展示的對(duì)象
        """
        # 首先調(diào)用父類的方法
        obj = super(ArticleDetailView, self).get_object()
        # 瀏覽量 +1
        obj.total_views += 1
        obj.save(update_fields=['total_views'])
        return obj

方法get_object()的作用是獲取需要展示的對(duì)象。首先調(diào)用父類方法,將這個(gè)對(duì)象賦值給obj變量,然后再對(duì)其進(jìn)行統(tǒng)計(jì)瀏覽量的操作,最后將對(duì)象返回。相當(dāng)于在原有的方法中把自己的邏輯“塞”了進(jìn)去。

關(guān)于DetailView更多特性看這里:官方文檔

編輯

除了能夠展示信息,通用視圖還包含CreateView、UpdateViewDeleteView編輯數(shù)據(jù)的類。

如果要新建文章,則視圖可以這么寫:

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'

創(chuàng)建create_by_class_view.html文件(目錄在哪,你應(yīng)該已經(jīng)很清楚了),寫入:

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='...'),
]

雖然外觀簡(jiǎn)陋(這不是重點(diǎn)),但現(xiàn)在這個(gè)視圖確實(shí)已經(jīng)能夠創(chuàng)建新文章了!

UpdateViewDeleteView這里就不再贅述了,以后用到的地方再進(jìn)行講解。

想提前了解的同學(xué)戳這里:官方文檔

總結(jié)

有沒有感受到代碼隔離繼承的強(qiáng)大?沒有?以后的章節(jié)會(huì)逐漸使用編寫視圖,你會(huì)慢慢體會(huì)的。

類視圖的內(nèi)容非常豐富,短短一篇文章只能蜻蜓點(diǎn)水而已。讀者在編程中遇到困難了,官方文檔是你最好的教程。

如果你有耐心從頭到尾閱讀類視圖的官方文檔,那當(dāng)然是最好的了。


  • 有疑問請(qǐng)?jiān)?a target="_blank" rel="nofollow">杜賽的個(gè)人網(wǎng)站留言,我會(huì)盡快回復(fù)。
  • 或Email私信我:dusaiphoto@foxmail.com
  • 項(xiàng)目完整代碼:Django_blog_tutorial

轉(zhuǎn)載請(qǐng)告知作者并注明出處。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 基于類的視圖 Django中的視圖是一個(gè)可調(diào)用對(duì)象,它接收一個(gè)請(qǐng)求然后返回一個(gè)響應(yīng)。這個(gè)可調(diào)用對(duì)象不僅僅限于函數(shù),...
    蘭山小亭閱讀 4,761評(píng)論 1 13
  • Django基于類的視圖 1.基于類的視圖簡(jiǎn)介 基于類的視圖使用Python 對(duì)象實(shí)現(xiàn)視圖,它提供除函數(shù)視圖之外的...
    常大鵬閱讀 8,736評(píng)論 0 25
  • 本教程內(nèi)容已過時(shí),更新版教程請(qǐng)?jiān)L問: Django 博客開發(fā)入門教程。 通過三周的時(shí)間我們開發(fā)了一個(gè)簡(jiǎn)單的個(gè)人 B...
    追夢(mèng)人物閱讀 6,927評(píng)論 6 14
  • 利用HTTP協(xié)議向服務(wù)器傳參的幾種途徑、響應(yīng)、Cookie、Session、類視圖、中間件 注意: 1>Dja...
    Cestine閱讀 1,492評(píng)論 0 2
  • 在給學(xué)生上課時(shí),做到一篇丁立梅的《小扇輕搖的時(shí)光》,里面有這樣一句“蟲鳴在四周此起彼伏地響起,南瓜花在夜里靜靜地開...
    江水1989閱讀 493評(píng)論 0 2

友情鏈接更多精彩內(nèi)容