美文网首页
Django -- Polls - Part 4

Django -- Polls - Part 4

作者: liaozb1996 | 来源:发表于2018-03-20 23:11 被阅读0次

Form

  • 如果是从服务器获取信息,使用 GET;如果是要从服务器修改信息,使用 POST;发生敏感信息,使用 POST
  • 上传文件,需要在 form 添加属性 enctype="multipart/form-data"

HTML Form

# polls/templates/polls/detail.html

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

{% if error_message %}<p style="color: red;"><strong>{{ error_message }}<strong></p>{% endif %}
    <form action="{% url 'polls:vote' question.id %}" method="post">
    {% csrf_token %} <!-- 防御跨站请求伪造,每个POST action指向同一个网站时必须包含 -->
    {% for choice in question.choice_set.all %}
    <input type="radio" id="choice{{ forloop.counter }}" name="choice" value="{{ choice.id }}" />
    <label for="choice{{ forloop.counter }}">{{choice.choice_text}}</label>
    {% endfor %}
    <br />
    <br />
    <input type="submit" />
</form>

处理投票的视图

# polls/views.py
def vote(request, question_id):
    question = get_object_or_404(Question, pk=question_id)
    try:
        selected_choice = get_object_or_404(Choice, pk=request.POST['choice']) # 类似字典,POST的值都是字符串
    except (KeyError, Choice.DoesNotExist):
        # KeyError: 当 POST 中不存在 choice
        error_message = 'Please select a choice!'
        context = {'question': question, 'error_message': error_message}
        return render(request, 'polls/detail.html', context)
    else:
        selected_choice.vote += 1
        selected_choice.save()
        return HttpResponseRedirect(reverse('polls:result', args=(question_id, )))
        # 每次成功处理 POST后都要进行重定向,避免用户点击 "后退" 时重复提交

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

投票加一的处理逻辑有一点问题:当两个用户同时投票,取得值 42,同时加一,保存到数据库的值会时43(正确应是 44)
【解决方法:Avoiding race conditions using F()


# polls/templates/polls/result.html

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

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

投票结果的模板

# polls/templates/polls/result.html

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

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

request and response documentation.

通用视图

开发Web通常会:根据URL从数据库获取数据,然后再渲染模板。
Django 提供了通用视图 generic 来简化步骤

修改 URL Pattern

# polls/urls.py
urlpatterns = [
    path('', views.IndexView.as_view(), name='index'),
    path('<int:pk>/', views.DetailView.as_view(), name='detail'),
    path('<int:pk>/result/', views.ResultView.as_view(), name='result'),
    path('<int:question_id>/vote/', views.vote, name='vote'),
]

使用通用视图

from django.views import generic
from .models import Question, Choice

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

    def get_queryset(self):
        return Question.objects.order_by('-pub_date')[:5]

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

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

分类

  • ListView: display a list of objects
  • DetailView: display a detail page for a particular type of object.

  • 每个通用视图都需要一个 model 变量
  • DetailView 需要从URL中获得 pk (primary key)

默认模板

  • ListView: <app name>/<model name>_list.html
  • DetailVew: <app name>/<model name>_detail.html
    可以通过变量 template_name 修改

默认 context

默认传递给模板的环境变量名

  • ListView: <model name>_list (小写的model name)
  • DetailView: <model name> (小写)
    可以通过变量 context_object_name attribute 修改

generic views documentation.

相关文章

  • Django -- Polls - Part 4

    Form 如果是从服务器获取信息,使用 GET;如果是要从服务器修改信息,使用 POST;发生敏感信息,使用 PO...

  • Django -- Polls - Part 2

    数据库配置 创建 Model Django Admin 本节包含的内容较多:Polls - Part 2 数据库配...

  • Django -- Polls - Part 3

    URL dispatcher View 视图 通过Python函数(或一个类的方法)来生成一个页面;可能会涉及通过...

  • Django -- Polls - Part 1

    创建项目结构 创建 APP -- Polls 结构 创建视图 View 配置/声明 URL -- URLconf ...

  • Django -- Polls - Part 7

    定制 Django Admin 步骤: 创建 admin.ModelAdmin 将子类作为 admin.site....

  • Django -- Polls - Part 6

    Django 渲染一个页面所需的图片,CSS, Javascript 称为静态文件 CSS 在APP下创建目录:p...

  • Django study

    [TOC] Django study creating a project 调试 demo-the Polls a...

  • django入门基础指令

    安装django指令 新建项目(名称为mysite) 运行django项目 创建应用(名称为polls) 为应用生...

  • 第一个Django App(六)

    静态文件 django.contrib.staticfiles 定制app的外观 在polls下创建static目...

  • Django学习(3)--管理站点

    编辑polls下面的admin.py进行站点管理。 加入了Question: from django.contri...

网友评论

      本文标题:Django -- Polls - Part 4

      本文链接:https://www.haomeiwen.com/subject/zociqftx.html