asp python3.0,Python3.0 Django2.0.4 搭建项目(三)

概况

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

博客首页——展示最近的几项内容。

内容“详情”页——详细展示某项内容。

以年为单位的归档页——展示选中的年份里各个月份创建的内容。

以月为单位的归档页——展示选中的月份里各天创建的内容。

以天为单位的归档页——展示选中天里创建的所有内容。

评论处理器——用于响应为一项内容添加评论的操作。

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

问题索引页——展示最近的几个投票问题。

问题详情页——展示某个投票的问题和不带结果的选项列表。

问题结果页——展示某个投票的结果。

投票处理器——用于响应用户为某个问题的特定选项投票的操作。

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

在你上网的过程中,很可能看见过像这样的 URL: "ME2/Sites/dirmod.asp?sid=&type=gen&mod=Core+Pages&gid=A6CD4967199A42D9B65B1B" 。在 Django 里的 URL 规则 要比这优雅的多!

一个 URL 模式定义了某种 URL 的基本格式——举个例子:/polls/20/vote/。

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

创建视图

在上一篇中,我们已经在polls/views.py文件下面创建了一个index的视图,接下来我们再创建几个简单的视图:

polls/views.py

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)

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

polls/urls.py

from django.urls import path

from . import views

urlpatterns = [

# ex: /polls/

path('', views.index, name='index'),

# ex: /polls/5/

path('/', views.detail, name='detail'),

# ex: /polls/5/results/

path('/results/', views.results, name='results'),

# ex: /polls/5/vote/

path('/vote/', views.vote, name='vote'),

]

配置完成之后我们在浏览器中查看,在浏览器中访问:http://127.0.0.1:8000/polls/3/

会看到如下结果:

8b9bc2970c4c

image.png

我们可以在浏览器访问以下以下两个地址,会发现跳转到我们预设的一个占位的投票页和结果页

8b9bc2970c4c

image.png

8b9bc2970c4c

image.png

创建一个简单业务逻辑视图

每个视图都会做两个事情:返回一个包含被请求页面内容的 HttpResponse 对象,或者抛出一个异常,比如Http404。

视图可以从数据库里读取记录,可以使用一个模板引擎(比如 Django 自带的,或者其他第三方的),可以生成一个 PDF 文件,可以输出一个 XML,创建一个 ZIP 文件,你可以做任何你想做的事,使用任何你想用的 Python 库。

我们在 index() 函数里插入了一些新内容,让它能展示数据库里以发布日期排序的最近 5 个投票问题:

polls/views.py

from django.http import HttpResponse

from .models import Question

def index(request):

latest_question_list = Question.objects.order_by('-pub_date')[:5]

output = ', '.join([q.question_text for q in latest_question_list])

return HttpResponse(output)

将下面的代码输入到刚刚创建的模板文件中:

polls/templates/polls/index.html

{% if latest_question_list %}

{% for question in latest_question_list %}

{{ question.question_text }}

{% endfor %}

{% else %}

No polls are available.

{% endif %}

注意事项:在这个view里面,页面的设计是直接写死在视图函数的代码里面。如果需要修改页面,需要编辑 Python 代码。 这时候我们可以使用Django的模板系统,只需要创建一个视图,就可以把页面的设计从代码中分离出来。

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

然后在 templates 目录里再创建一个目录 polls,然后在其中新建一个文件 index.html。此时模板文件的路径应该是 polls/templates/polls/index.html 。Django 会寻找到对应的 app_directories ,所以只需要使用 polls/index.html 就可以引用到这一模板了。

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

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))

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

用你的浏览器访问 "http://127.0.0.1:8000/polls/" ,你将会看见一个无序列表,列出了我们在第2部分 中添加的 “What's up” 投票问题,链接指向这个投票的详情页。

8b9bc2970c4c

image.png

点击What's up? 选项效果如下:

8b9bc2970c4c

image.png

函数

1.render()

载入模板,填充上下文,再返回由它生成的 HttpResponse 对象,是一个非常常用的操作流程。于是 Django 提供了一个快捷函数,我们用它来重写 index() 视图:

polls/views.py

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)

注意到,我们不再需要导入 loader 和 HttpResponse 。

抛出 404 错误 get_object_or_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})

这里我们设置了一个规则,如果指定问题 ID 所对应的问题不存在,这个视图就会抛出一个 Http404 异常。

接下来我们通过get_object_or_404()函数来实现:

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})

接下来我们完善一下detail.html模板:

polls/templates/polls/detail.html

{{ question.question_text }}

{% for choice in question.choice_set.all %}

{{ choice.choice_text }}

{% endfor %}

模板系统统一使用点符号来访问变量的属性。在示例 {{ question.question_text }} 中,首先 Django 尝试对 question 对象使用字典查找(也就是使用 obj.get(str) 操作),如果失败了就尝试属性查找(也就是 obj.str 操作),结果是成功了。如果这一操作也失败的话,将会尝试列表查找(也就是 obj[int] 操作)。

在 {% for %} 循环中发生的函数调用:question.choice_set.all 被解释为 Python 代码 question.choice_set.all() ,将会返回一个可迭代的 Choice 对象,这一对象可以在 {% for %} 标签内部使用。

去除模板中的硬编码 URL

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

{{ question.question_text }}

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

{{ question.question_text }}

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

# added the word 'specifics'

path('specifics//', 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('/', views.detail, name='detail'),

path('/results/', views.results, name='results'),

path('/vote/', views.vote, name='vote'),

]

现在,编辑 polls/index.html 文件,将以下:

polls/templates/polls/index.html

{{ question.question_text }}

修改为:

polls/templates/polls/index.html

{{ question.question_text }}

到这里,我们已经能完整的做一个视图,并且和模板一起使用。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值