尊重作者的勞動,轉載請註明作者及原文地址 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/
本節完