zoukankan      html  css  js  c++  java
  • python 3+djanjo 2.0.7简单学习(四)--Django视图

    1.概念

    Django 中的视图的概念是「一类具有相同功能和模板的网页的集合」。比如,在一个博客应用中,你可能会创建如下几个视图:

    • 博客首页——展示最近的几项内容。
    • 内容“详情”页——详细展示某项内容。
    • 以年为单位的归档页——展示选中的年份里各个月份创建的内容。
    • 以月为单位的归档页——展示选中的月份里各天创建的内容。
    • 以天为单位的归档页——展示选中天里创建的所有内容。
    • 评论处理器——用于响应为一项内容添加评论的操作。

    而在我们的投票应用中,我们需要下列几个视图:

    • 问题索引页——展示最近的几个投票问题。
    • 问题详情页——展示某个投票的问题和不带结果的选项列表。
    • 问题结果页——展示某个投票的结果。
    • 投票处理器——用于响应用户为某个问题的特定选项投票的操作。

    在 Django 中,网页和其他内容都是从视图派生而来。每一个视图表现为一个简单的 Python 函数(或者说方法,如果是在基于类的视图里的话)。Django 将会根据用户请求的 URL 来选择使用哪个视图(更准确的说,是根据 URL 中域名之后的部分)

    为了将 URL 和视图关联起来,Django 使用了 'URLconfs' 来配置。URLconf 将 URL 模式映射到视图。

    2.添加视图:

    votes/views.py

    from
    django.shortcuts import render,HttpResponse # Create your views here. def index(request): return HttpResponse("Hello world!") def detail(request,question_id): return HttpResponse("You're looking at question %s." % question_id) def results(request, question_id): response = "You're looking at the results of question %s." return HttpResponse(response % question_id) def vote(request, question_id): return HttpResponse("You're voting on question %s." % question_id)

    把这些新视图添加进 votes.urls 模块里,只要添加几个 url() 函数调用就行:

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

    然后看看你的浏览器,如果你转到 "/votes/1/" ,Django 将会运行 detail() 方法并且展示你在 URL 里提供的问题 ID,如下:。

    再试试 "/votes/1/results/" 和 "/votes/1/vote/" ——你将会看到暂时用于占位的结果和投票页。

    当某人请求你网站的某一页面时——比如说, "/votes/1/" ,Django 将会载入 mysite.urls 模块,因为这在配置项 ROOT_URLCONF 中设置了。然后 Django 寻找名为 urlpatterns 变量并且按序匹配正则表达式。在找到匹配项 'votes/',它切掉了匹配的文本("votes/"),将剩余文本——"1/",发送至 'votes.urls' URLconf 做进一步处理。在这里剩余文本匹配了 '<int:question_id>/',使得我们 Django 以如下形式调用 detail():

    detail(request=<HttpRequest object>, question_id=1)

    question_id=1由 <int:question_id> 匹配生成。使用尖括号“捕获”这部分 URL,且以关键字参数的形式发送给视图函数。上述字符串的 :question_id> 部分定义了将被用于区分匹配模式的变量名,而 int: 则是一个转换器决定了应该以什么变量类型匹配这部分的 URL 路径。

    为每个 URL 加上不必要的东西,例如 .html ,是没有必要的。不过如果你非要加的话,也是可以的:

    path('votes/latest.html', views.index),

     但是,别这样做,这太傻了

    3.接下来写一个真正的视图

    这里有个问题:页面的设计写死在视图函数的代码里的。如果你想改变页面的样子,你需要编辑 Python 代码。所以让我们使用 Django 的模板系统,只要创建一个视图,就可以将页面的设计从代码中分离出来。

    首先,在你的 votes 目录里创建一个 templates 目录。Django 将会在这个目录里查找模板文件。

    你项目的 TEMPLATES 配置项描述了 Django 如何载入和渲染模板。默认的设置文件设置了 DjangoTemplates 后端,并将 APP_DIRS 设置成了 True。这一选项将会让 DjangoTemplates 在每个 INSTALLED_APPS 文件夹中寻找 "templates" 子目录。这就是为什么尽管我们没有像在第二部分中那样修改 DIRS 设置,Django 也能正确找到 votes 的模板位置的原因。

    在你刚刚创建的 templates 目录里,再创建一个目录 votes,在其中新建一个文件 index.html 。换句话说,你的模板文件的路径应该是 votes/templates/votes/index.html 。因为 Django 会寻找到对应的 app_directories ,所以你只需要使用votes/index.html 就可以引用到这一模板了。

    votes/templates/votes/index.html
    
    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>首页</title>
    </head>
    <body>
        {% if latest_question_list %}
        <ul>
        {% for question in latest_question_list %}
            <li><a href="/votes/{{ question.id }}/">{{ question.question_text }}</a></li>
        {% endfor %}
        </ul>
        {% else %}
            <p>No votes are available.</p>
        {% endif %}
    
    </body>
    </html>

    然后,让我们更新一下 votes/views.py 里的 index 视图来使用模板:

    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('votes/index.html')
        context = {
            'latest_question_list': latest_question_list,
        }
        return HttpResponse(template.render(context, request))

    上述代码的作用是,载入 votes/index.html 模板文件,并且向它传递一个上下文(context)。这个上下文是一个字典,它将模板内的变量映射为 Python 对象。

    这里可以优化一下使用render函数()

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

    简洁了很多

    接下来编写投票详情视图

    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, 'votes/detail.html', {'question': question})

    html

    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>问题详情</title>
    </head>
    <body>
        <h1>{{ question.question_text }}</h1>
        <ul>
            {% for choice in question.choice_set.all %}
        <li>{{ choice.choice_text }}</li>
            {% endfor %}
        </ul>
    
    </body>
    </html>

    4.去除模板中的硬编码 URL

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

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

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

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

    这个标签的工作方式是在 votes.urls 模块的 URL 定义中寻具有指定名字的条目。你可以回忆一下,具有名字 'detail' 的 URL 是在如下语句中定义的:

    ...
    # the 'name' value as called by the {% url %} template tag
    path('<int:question_id>/', views.detail, name='detail'),
    ...

    如果你想改变投票详情视图的 URL,比如想改成 votes/specifics/12/ ,你不用在模板里修改任何东西(包括其它模板),只要在 votes/urls.py 里稍微修改一下就行:

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

    5.为 URL 名称添加命名空间

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

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

    from django.urls import path
    from . import views
    app_name='votes'
    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'),
    ]

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

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

    修改为指向具有命名空间的详细视图:

    <li><a href="{% url 'votes:detail' question.id %}">{{ question.question_text }}</a></li>
     
    目前还在学习中,希望会对大家有所帮助,觉得不错,就点赞支持一下。 另外,转载时请附带链接。谢谢!
  • 相关阅读:
    Atitit 图片验证码功能设计文档总结目录1.1. 使用图片验证码img src标签设置图片。。验证码图片有png,jpg,svg等格式。。 11.2. Php png图像 11.3. P
    Atitt php script lan debug bp 最佳实践调试php目录1.1. Error_log 11.2. Echo vs log法 11.3. 输出与debug信息昏药问题
    Atitit 项目分析与统计目录1. 静态分析+动态分析 。其中, 12. 模块分析,与模块位置idx 13. 编程语言类型与版本 13.1. 类库统记表 类型与版本 23.2. 中间
    Atitit 增强代码健壮性 出错继续执行恢复模式,就像vbs那样我以为我可以使用Try/Catch,但是我找不到异常后是否可以继续执行代码,并且找不到如何在最后显示错误消息。目录PHP
    Atitit bootsAtitit bootstrap布局 栅格.docx目录1. 简述container与container-fluid的区别 11.1.1. 在bootstrap中的布局
    Atitit php读取数据库记录集合并循环修改展示//------------------------ini db sys$dbstr = “mysql:host=“ . $mysql_con
    Atitit 验证码功能修复总结文档原有的tp5里面的验证码不知怎么有问题了,试图在tp5框架内修复无果。。使用了新的验证码组件 “lifei6671/php-captcha“: “0.
    Atitt 支付业务 银行国际代码(SWIFT Code银行国际代码(SWIFT Code)是由SWIFT协会提出并被ISO通过的银行识别代码,凡该协会的成员银行都有自己特定的SWIFT代码
    Atitit img hot click link 图像背景拉伸100%Map area trouble..So body backgrd img mode...is easy...No
    Atitit doc mng 文档管理总结目录1. 主要几大内容 12. 存储管理 22.1. 一般来说 ,文档存储在IM网盘note邮箱blog wiki等地 22.2. 文档格式与体
  • 原文地址:https://www.cnblogs.com/dangkai/p/9548704.html
Copyright © 2011-2022 走看看