![Django 3.0入门与实践](https://wfqqreader-1252317822.image.myqcloud.com/cover/246/36862246/b_36862246.jpg)
3.10 添加视图
现在网站的后台管理模块已经可以工作了,还缺少前台页面。投票系统需要以下几个页面:
问题索引页——展示最近的几个投票问题。
问题详情页——展示某个投票的问题和不带结果的选项列表。
问题结果页——展示某个投票的结果。
投票处理器——用于响应用户为某个问题的特定选项投票的操作。
在Django中每一个页面或者其他内容都是通过视图呈现出来的,每一个视图就是一个Python函数或者类方法,Django中的视图是“一类具有相同功能和模板的网页的集合”。Django通过URL确定调用哪一个视图,Django的URL相较于早期网站的URL更加简洁优雅。
Django通过URLconfs将URL模式字符串与视图关联起来,URL模式字符串就是一个URL的一般形式,如/newsarchive/<year>/<month>/。
在polls/views.py文件中添加以下视图:
![](https://epubservercos.yuewen.com/B89DD0/19549641008916106/epubprivate/OEBPS/Images/Figure-P30_149232.jpg?sign=1738940621-wLu5KR8OZwb3HtryB5itMXO0Z9X3Fse1-0-a58e5a970fe19b059e6e07837551b412)
修改polls.urls文件,添加以下URL映射:
![](https://epubservercos.yuewen.com/B89DD0/19549641008916106/epubprivate/OEBPS/Images/Figure-P30_149234.jpg?sign=1738940621-Maua2b8RWfDLgBoUq3JIQyrVLSLslCnP-0-60fe18dc11622f1807a0d4263cdc9d5a)
好了,重启Web服务器,在浏览器中访问http://127.0.0.1:8000/polls/24/,如下图所示。
![](https://epubservercos.yuewen.com/B89DD0/19549641008916106/epubprivate/OEBPS/Images/Figure-P30_15499.jpg?sign=1738940621-Ba1KDRnOD6J5rLVscJQXH4Zok9c1OvqU-0-513e6387ebd231bfad2cc42163f6d28a)
继续访问http://127.0.0.1:8000/polls/24/results/和http://127.0.0.1:8000/polls/24/vote/,同样能够正常显示视图内容,如下面两幅图所示。
![](https://epubservercos.yuewen.com/B89DD0/19549641008916106/epubprivate/OEBPS/Images/Figure-P31_15937.jpg?sign=1738940621-8Xm32Rcelli7DwlyRuep6Bq6SyYxtDsM-0-8b34ff26273a39ddb6bb8cc85c2f42c5)
![](https://epubservercos.yuewen.com/B89DD0/19549641008916106/epubprivate/OEBPS/Images/Figure-P31_15938.jpg?sign=1738940621-IE0ePhwVyaVNVBlcgOStHtTHL3tI4kF3-0-dbf1c2ffa466e1da13bdc446f653f397)
之所以Django能够正常调用解析URL,是因为在settings.py中设置了ROOT_URLCONF = 'mysite.urls'。当用户访问的URL包含polls/时,Django会根据mysite.urls中的设置,跳转到polls.urls并进行验证,直到找到第一个匹配的URL为止。
以上视图中参数question_id的值来自于<int:question_id>。<int:question_id>用于匹配URL中的值,并将捕捉到的值作为关键字参数传递给视图,其中:question_id对应视图的参数,int:决定了URL中的哪类值符合匹配条件。
3.10.1 扩展视图
每一个视图都应该负责一个具体的业务逻辑,视图执行结束会返回一个包含页面内容的HttpResponse对象或者异常信息。
下面修改index视图使它返回最新的5条调查问卷。
![](https://epubservercos.yuewen.com/B89DD0/19549641008916106/epubprivate/OEBPS/Images/Figure-P31_149236.jpg?sign=1738940621-cx7odpF6PQHUhB51KJ8Zvz9dB5TdbUb2-0-ae0164e050c5fc869aca9c71f0dbb95b)
代码Question.objects.order_by('-pub_date')是Django的数据库API语法,用于从数据库中查找数据,在介绍模型时将进行详细讲解。
访问index页面以查看显示情况,如下图所示。
![](https://epubservercos.yuewen.com/B89DD0/19549641008916106/epubprivate/OEBPS/Images/Figure-P32_16405.jpg?sign=1738940621-G63E68kmEiLb9opvyyl2VCCTTgOZznNp-0-3b4cd31ccc2e4bd22613fa9fae678beb)
此时调查问卷已经显示到网页上,但是可以发现在index视图中使用了硬编码,如果想要修改网页显示样式就需要重新编写Python代码。对此Django提供了一套模板系统(templates),可以将业务逻辑与页面显示样式分离开。下面来看看如何使用模板系统。
首先在polls文件夹下创建一个新文件夹templates,为了目录结构清晰,在templates文件夹下再创建一个polls文件夹,最后在polls下创建一个index.html文件。这个index.html就是即将应用于index视图的模板。
在settings.py中有一个关于模板的配置项:TEMPLATES。Django就是根据这个配置查找并解析模板的,具体工作原理会在第5章进行讲解。
将下面代码写入模板文件index.html:
![](https://epubservercos.yuewen.com/B89DD0/19549641008916106/epubprivate/OEBPS/Images/Figure-P32_149238.jpg?sign=1738940621-Z9eSJcRNmfleKenbLa53KDPn5c7BufcZ-0-a09fa7cbb9cd27949e3663fc0b37493e)
接下来修改index视图:
![](https://epubservercos.yuewen.com/B89DD0/19549641008916106/epubprivate/OEBPS/Images/Figure-P32_149240.jpg?sign=1738940621-MC6rxhke3iIYeobhisb66TLThSThdpZr-0-cd1ee8ca3995791480d8eef41daa0b76)
新视图会从模板文件夹下加载模板文件并将一个字典对象传入视图。
重启Web服务器,刷新index页面,效果如下图所示。
![](https://epubservercos.yuewen.com/B89DD0/19549641008916106/epubprivate/OEBPS/Images/Figure-P33_16866.jpg?sign=1738940621-5P3lTge554ApT09DyIuJadHvZlS3ome7-0-ad092357b78f7194aa4c0645ee0b9e3a)
上面代码的工作原理是先使用loader方法加载模板文件并向它传递一个上下文对象(context),然后使用HttpResponse方法初始化一个HttpResponse对象并返回给浏览器。由于很多Django视图都是这样工作的,因此Django提供了一个简写函数:render()。下面使用render函数重写index视图:
![](https://epubservercos.yuewen.com/B89DD0/19549641008916106/epubprivate/OEBPS/Images/Figure-P33_149247.jpg?sign=1738940621-tVIpmlJgbx4V8ArZ2OAncn9jAzuq2X4S-0-2812e0b39250cd184de92eed54cbbd8f)
此时重新访问index页面时可以发现效果与之前一样,但是不再需要loader和HttpResponse方法。
3.10.2 处理404错误
404错误是一个比较常见的网页访问错误,当被访问的URL资源不存在时就会抛出这类错误。下面修改detail视图使其在无法查找到问卷的时候抛出404错误。
![](https://epubservercos.yuewen.com/B89DD0/19549641008916106/epubprivate/OEBPS/Images/Figure-P33_149245.jpg?sign=1738940621-2UkvBe7aTUdTbpjjwBb9yDadQsoPQR7F-0-a57eaef24810ef0a597ee3f88fae66ed)
按照前面步骤在polls文件夹下创建一个detail.html文件作为detail视图的模板文件,模板内容暂时用{{ question }}表示。
此时重启Web服务,分别访问一个存在的和一个不存在的问卷,效果如下面两幅图所示。
![](https://epubservercos.yuewen.com/B89DD0/19549641008916106/epubprivate/OEBPS/Images/Figure-P33_16872.jpg?sign=1738940621-Cxx2RWowqrEidTLdpE9PzMcMu9iHjmT3-0-0ed99fa9d3e5485a25dfa9a3996bd9cf)
![](https://epubservercos.yuewen.com/B89DD0/19549641008916106/epubprivate/OEBPS/Images/Figure-P34_17312.jpg?sign=1738940621-Zc0qn61YlNcFoNORHgg4Jjj8irs5oUh5-0-137330ab8f4e64341d406283ddcb5b14)
由于404错误是一个非常常见的网页异常,因此Django也提供了一个简写方法:get_object_or_404。下面使用get_object_or_404()修改detail视图:
![](https://epubservercos.yuewen.com/B89DD0/19549641008916106/epubprivate/OEBPS/Images/Figure-P34_149249.jpg?sign=1738940621-xyTVpkG2W5HowTitWUCJ0zqm4uCucMLl-0-419cb4c33dfedaec8ae3dcc70d726401)
重新访问detail页面,效果如下图所示。
![](https://epubservercos.yuewen.com/B89DD0/19549641008916106/epubprivate/OEBPS/Images/Figure-P34_17316.jpg?sign=1738940621-qxjPxKMSphPoWD8oEIrGuwpvYaQv1QLa-0-7973ed415622fc3c15a7df7f0f82024d)
此时网页仍然抛出404错误,不过错误信息变成Django默认的英文形式,此时可以通过修改get_object_or_404()方法源代码的方式修改错误信息,修改完需要重启Web服务。
与get_object_or_404相似,Django还提供了一个判断list是否存在的方法:get_list_or_404,在此不做详细介绍。