django入門-表單-part4

曲珂發表於2017-03-07

尊重作者的勞動,轉載請註明作者及原文地址 http://www.cnblogs.com/txwsqk/p/6514113.html 

完全翻譯自官方文件 https://docs.djangoproject.com/en/1.10/intro/tutorial04/

本節內容講表單

讓我們更新一下pools/detail.html,新增html表單的展示

<h1>{{ question.question_text }}</h1>

{% if error_message %}<p><strong>{{ error_message }}</strong></p>{% endif %}

<form action="{% url 'polls:vote' question.id %}" method="post">
{% csrf_token %}
{% for choice in question.choice_set.all %}
    <input type="radio" name="choice" id="choice{{ forloop.counter }}" value="{{ choice.id }}" />
    <label for="choice{{ forloop.counter }}">{{ choice.choice_text }}</label><br />
{% endfor %}
<input type="submit" value="Vote" />
</form>

解釋一下上面的程式碼:

form裡新增了一個radio單選框,label是單選框的標籤,每個單選框的值是 choice.id,當你點了"submit"提交表單時,實際提交的內容是

一個變數choice=choice.id 即 input的name和value,表單提交的方法必須是post

forloop.counte是for迴圈的一個計數器,表示迴圈了多少次

{% csrf_token %}  這個是用來防止跨站攻擊的,django已經幫我們做好了,放在form上就可以了

提交這個表單時,匹配的url是這個

url(r'^(?P<question_id>[0-9]+)/vote/$', views.vote, name='vote'),

現在更新一下我們的vote的檢視

from django.shortcuts import get_object_or_404, render
from django.http import HttpResponseRedirect, HttpResponse
from django.urls import reverse

from .models import Choice, Question
# ...
def vote(request, question_id):
    question = get_object_or_404(Question, pk=question_id)
    try:
        selected_choice = question.choice_set.get(pk=request.POST['choice'])
    except (KeyError, Choice.DoesNotExist):
        # Redisplay the question voting form.
        return render(request, 'polls/detail.html', {
            'question': question,
            'error_message': "You didn't select a choice.",
        })
    else:
        selected_choice.votes += 1
        selected_choice.save()
        # Always return an HttpResponseRedirect after successfully dealing
        # with POST data. This prevents data from being posted twice if a
        # user hits the Back button.
        return HttpResponseRedirect(reverse('polls:results', args=(question.id,)))

解釋request.POST 你用post提交表單時傳的引數就是用這個接收,是一個字典型別的;對應的request.GET就是接收get請求的引數

request.POST['choice']當key不存在時,會拋一個KeyError異常,所以程式碼要捕獲這個異常
HttpResponseRedirect(url)當成功提交表單返回結果時就用這個函式
reverse這個函式是為了避免url的硬編碼,本來這個url應該是這樣的

url(r'^(?P<question_id>[0-9]+)/results/$', views.results, name='results'),
現在呢
reverse('polls:results', args=(question.id,) 看不明白的請去複習上一節的內容(urls.py裡的app_name和url()裡的name就是這個用處

 

現在來完成result的檢視

from django.shortcuts import get_object_or_404, render


def results(request, question_id):
    question = get_object_or_404(Question, pk=question_id)
    return render(request, 'polls/results.html', {'question': question})

result的檢視有了,下面寫展示的模板

<h1>{{ question.question_text }}</h1>

<ul>
{% for choice in question.choice_set.all %}
    <li>{{ choice.choice_text }} -- {{ choice.votes }} vote{{ choice.votes|pluralize }}</li>
{% endfor %}
</ul>

<a href="{% url 'polls:detail' question.id %}">Vote again?</a>

這個例子就是你選擇單選按鈕,然後點選提交按鈕,跳轉到一個新的頁面,沒有使用ajax.

注意:

vote檢視這裡有併發問題,當多個人對同一個question投票時,資料庫在存和取時就有資源競爭問題了

django的解決方案在此 https://docs.djangoproject.com/en/1.10/ref/models/expressions/#avoiding-race-conditions-using-f

厲害的來了

通用檢視

考慮一下我們前面寫的檢視detail(),results(),index()都是web開發中最通用的操作,根據url的引數從資料庫存取內容,然後模板展示,這就是django通用檢視考慮的問題,下面讓我們用通用檢視修改一下我們前面的應用pools

先修改urls.py

from django.conf.urls import url

from . import views

app_name = 'polls'
urlpatterns = [
    url(r'^$', views.IndexView.as_view(), name='index'),
    url(r'^(?P<pk>[0-9]+)/$', views.DetailView.as_view(), name='detail'),
    url(r'^(?P<pk>[0-9]+)/results/$', views.ResultsView.as_view(), name='results'),
    url(r'^(?P<question_id>[0-9]+)/vote/$', views.vote, name='vote'),
]

 修改檢視views.py

from django.shortcuts import get_object_or_404, render
from django.http import HttpResponseRedirect
from django.urls import reverse
from django.views import generic

from .models import Choice, Question


class IndexView(generic.ListView):
    template_name = 'polls/index.html'
    context_object_name = 'latest_question_list'

    def get_queryset(self):
        """Return the last five published questions."""
        return Question.objects.order_by('-pub_date')[:5]


class DetailView(generic.DetailView):
    model = Question
    template_name = 'polls/detail.html'


class ResultsView(generic.DetailView):
    model = Question
    template_name = 'polls/results.html'


def vote(request, question_id):
    ... # same as above, no changes needed.

這裡我們用了兩個新的檢視ListView和DetailView

ListView 展示成列表

DetailView 根據不同的model展示不同的詳情頁

這兩個檢視都需要一個叫model的屬性,它得知道根據那個model展示相應的內容

DetailView通用檢視是根據url中的的pk引數來獲取資料庫內容的,所以我們urls.py做了相應修改

這兩個通用檢視如果你不傳template_name那麼會使用預設的模板檔案

DetailView使用<app name>/<modelname>_detail.html 本應用中就是 "polls/question_detail.html"

ListView使用<app name>/<modelname>_list.html ,當然你傳了template_name它就用你定義的模板了

 

在前面的章節中我們可以給urls.py裡定義的url()定義一個context字典來給view()傳引數,通用檢視有預定義的變數可以使用

DetailView使用model的小寫作為context變數的key

ListView使用model的小寫+_list作為context的key

當然你想使用自己的context可以傳context_object_name來自定義你的context

 

更多通用檢視的內容請參考 https://docs.djangoproject.com/en/1.10/topics/class-based-views/

 

本節完

相關文章