美文网首页
写一个真正有用的视图

写一个真正有用的视图

作者: 黑夜的眸 | 来源:发表于2018-05-05 22:43 被阅读0次

    载入模板,填充上下文,再返回由它生成的 HttpResponse 对象

    是一个非常常用的操作流程。

    polls/views.py

    from django.http import HttpResponse
    from django.template import loader
    
    from .models import Question
    
    
    def index(request):
        latest_question_list = Question.objects.order_by('-pub_date')[:5]
        template = loader.get_template('polls/index.html')
        context = {
            'latest_question_list': latest_question_list,
        }
        return HttpResponse(template.render(context, request))
    

    Django 提供了一个快捷函数render(),我们用它来重写 index() 视图:

    from django.shortcuts import render
    
    from .models import Question
    
    
    def index(request):
        latest_question_list = Question.objects.order_by('-pub_date')[:5]
        context = {'latest_question_list': latest_question_list}
        return render(request, 'polls/index.html', context)
    

    这里有个问题:页面的设计写死在视图函数的代码里的。如果你想改变页面的样子,你需要编辑 Python 代码。所以让我们使用 Django 的模板系统,只要创建一个视图,就可以将页面的设计从代码中分离出来。
    首先,在你的 polls 目录里创建一个 templates 目录。Django 将会在这个目录里查找模板文件。
    你项目的 TEMPLATES 配置项描述了 Django 如何载入和渲染模板。默认的设置文件设置了 DjangoTemplates 后端,并将 APP_DIRS 设置成了 True。这一选项将会让 DjangoTemplates 在每个 INSTALLED_APPS 文件夹中寻找 "templates" 子目录。这就是为什么尽管我们没有像在第二部分中那样修改 DIRS 设置,Django 也能正确找到 polls 的模板位置的原因。
    在你刚刚创建的 templates 目录里,再创建一个目录** polls**,然后在其中新建一个文件 index.html 。换句话说,你的模板文件的路径应该是 polls/templates/polls/index.html 。因为 Django 会寻找到对应的 app_directories ,所以你只需要使用 polls/index.html 就可以引用到这一模板了。
    另外,latest_question_list = Question.objects.order_by('-pub_date')[:5]
    是筛选出最近的5个问题,涉及到数据库操作,如果要使用原生数据库语言,则是:

    latest_question_list = Question.objects.raw('SELECT * FROM polls_question ORDER BY pub_date')[:5]
    

    抛出 404 错误

    现在,我们来处理投票详情视图——它会显示指定投票的问题标题。下面是这个视图的代码:

    polls/views.py

    from django.http import Http404
    from django.shortcuts import render
    
    from .models import Question
    
    def detail(request, question_id):
        try:
            question = Question.objects.get(pk=question_id)
        except Question.DoesNotExist:
            raise Http404("Question does not exist")
        return render(request, 'polls/detail.html', {'question': question})
    

    Django 也提供了一个快捷函数 get_object_or_404(),下面是修改后的详情 detail() 视图代码:
    polls/views.py

    from django.shortcuts import get_object_or_404, render
    
    from .models import Question
    # ...
    def detail(request, question_id):
        question = get_object_or_404(Question, pk=question_id)
        return render(request, 'polls/detail.html', {'question': question})
    

    下面是 polls/detail.html 模板里正式的代码:

    polls/templates/polls/detail.html

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

    去除模板中的硬编码 URL

    还记得吗,我们在 polls/index.html 里编写投票链接时,链接是硬编码的:

    <li><a href="/polls/{{ question.id }}/">{{ question.question_text }}</a></li>
    

    问题在于,硬编码和强耦合的链接,对于一个包含很多应用的项目来说,修改起来是十分困难的。然而,因为你在 polls.urls 的 url() 函数中通过 name 参数为 URL 定义了名字,你可以使用 {% url %} 标签代替它:

    <li><a href="{% url 'detail' question.id %}">{{ question.question_text }}</a></li>
    

    这个标签的工作方式是在 polls.urls 模块的 URL 定义中寻具有指定名字的条目。
    如果你想改变投票详情视图的 URL,比如想改成 polls/specifics/12/ ,你不用在模板里修改任何东西(包括其它模板),只要在 polls/urls.py 里稍微修改一下就行:

    # added the word 'specifics'
    path('specifics/<int:question_id>/', views.detail, name='detail'),
    

    为 URL 名称添加命名空间

    教程项目只有一个应用,polls 。在一个真实的 Django 项目中,可能会有五个,十个,二十个,甚至更多应用。Django 如何分辨重名的 URL 呢?举个例子,polls 应用有 detail 视图,可能另一个博客应用也有同名的视图。Django 如何知道 {% url %} 标签到底对应哪一个应用的 URL 呢?

    答案是:在根 URLconf 中添加命名空间。在 polls/urls.py 文件中稍作修改,加上 app_name 设置命名空间:

    polls/urls.py

    from django.urls import path
    
    from . import views
    
    app_name = 'polls'
    urlpatterns = [
        path('', views.index, name='index'),
        path('<int:question_id>/', views.detail, name='detail'),
        path('<int:question_id>/results/', views.results, name='results'),
        path('<int:question_id>/vote/', views.vote, name='vote'),
    ]
    

    现在,编辑 polls/index.html 文件,从:

    polls/templates/polls/index.html

    <li><a href="{% url 'detail' question.id %}">{{ question.question_text }}</a></li>
    

    修改为指向具有命名空间的详细视图:
    polls/templates/polls/index.html

    <li><a href="{% url 'polls:detail' question.id %}">{{ question.question_text }}</a></li>
    

    相关文章

      网友评论

          本文标题:写一个真正有用的视图

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