设为首页 收藏本站
查看: 1361|回复: 0

[经验分享] Python框架之Django学习笔记(十七)

[复制链接]

尚未签到

发表于 2015-4-20 13:24:40 | 显示全部楼层 |阅读模式
  Django框架之表单(续二)
  今天的这篇博客将是Django学习笔记博客的最后一篇,基本每周最少一篇的Django框架学习,坚持到今天也实属不易,当然了,这个框架的学习仅仅是Django框架的基础部分了,不过也够我们平时搭个简易的网站或者个人博客什么的。希望通过这一系列的博文,让大家也从中体会到Django框架的魅力所在,如果很不幸,你没有体会到,只能说明我水平有限,无法将如此美丽的事物展示与你,闲话少说,下面开始继续学习表单的相关知识。
  编写Contact表单
  这个表单包括用户提交的反馈信息,一个可选的e-mail回信地址。 当这个表单提交并且数据通过验证后,系统将自动发送一封包含题用户提交的信息的e-mail给站点工作人员。


  我们从contact_form.html模板入手:




1
2
3     Contact us
4
5
6     Contact us
7
8     {% if errors %}
9         
10             {% for error in errors %}
11             {{ error }}
12             {% endfor %}
13         
14     {% endif %}
15
16     
17         Subject:
18         Your e-mail (optional):
19         Message:
20         
21     
22
23
  我们定义了三个字段: 主题,e-mail和反馈信息。 除了e-mail字段为可选,其他两个字段都是必填项。 注意,这里我们使用method=”post”而非method=”get”,因为这个表单会有一个服务器端的操作:发送一封e-mail。 并且,我们复制了前一个模板search_form.html中错误信息显示的代码。
  如果我们顺着上一节编写search()视图的思路,那么一个contact()视图代码应该像这样:





1 from django.core.mail import send_mail
2 from django.http import HttpResponseRedirect
3 from django.shortcuts import render_to_response
4
5 def contact(request):
6     errors = []
7     if request.method == 'POST':
8         if not request.POST.get('subject', ''):
9             errors.append('Enter a subject.')
10         if not request.POST.get('message', ''):
11             errors.append('Enter a message.')
12         if request.POST.get('email') and '@' not in request.POST['email']:
13             errors.append('Enter a valid e-mail address.')
14         if not errors:
15             send_mail(
16                 request.POST['subject'],
17                 request.POST['message'],
18                 request.POST.get('email', 'noreply@example.com'),
19                 ['siteowner@example.com'],
20             )
21             return HttpResponseRedirect('/contact/thanks/')
22     return render_to_response('contact_form.html',
23         {'errors': errors})
  现在来分析一下以上的代码:


  • 确认request.method的值是’POST’。用户浏览表单时这个值并不存在,当且仅当表单被提交时这个值才出现。 (在后面的例子中,request.method将会设置为’GET’,因为在普通的网页浏览中,浏览器都使用GET,而非POST)。判断request.method的值很好地帮助我们将表单显示与表单处理隔离开来。
  • 我们使用request.POST代替request.GET来获取提交过来的数据。 这是必须的,因为contact_form.html里表单使用的是method=”post”。如果在视图里通过POST获取数据,那么request.GET将为空。
  • 这里,有两个必填项,subject 和 message,所以需要对这两个进行验证。 注意,我们使用request.POST.get()方法,并提供一个空的字符串作为默认值;这个方法很好的解决了键丢失与空数据问题。


  • 虽然email非必填项,但如果有提交她的值则我们也需进行验证。 我们的验证算法相当的薄弱,仅验证值是否包含@字符。 在实际应用中,需要更为健壮的验证机制(Django提供这些验证机制,稍候我们就会看到)。
  • 我们使用了django.core.mail.send_mail函数来发送e-mail。 这个函数有四个必选参数: 主题,正文,寄信人和收件人列表。 send_mail是Django的EmailMessage类的一个方便的包装,EmailMessage类提供了更高级的方法,比如附件,多部分邮件,以及对于邮件头部的完整控制。
  • 注意,若要使用send_mail()函数来发送邮件,那么服务器需要配置成能够对外发送邮件,并且在Django中设置出站服务器地址。参见规范:http://docs.djangoproject.com/en/dev/topics/email/
  • 当邮件发送成功之后,我们使用HttpResponseRedirect对象将网页重定向至一个包含成功信息的页面。 包含成功信息的页面这里留给读者去编写(很简单 一个视图/URL映射/一份模板即可),但是我们要解释一下为何重定向至新的页面,而不是在模板中直接调用render_to_response()来输出。
  • 原因就是: 若用户刷新一个包含POST表单的页面,那么请求将会重新发送造成重复。 这通常会造成非期望的结果,比如说重复的数据库记录;在我们的例子中,将导致发送两封同样的邮件。 如果用户在POST表单之后被重定向至另外的页面,就不会造成重复的请求了。
  • 我们应每次都给成功的POST请求做重定向。 这就是web开发的最佳实践。
  contact()视图可以正常工作,但是她的验证功能有些复杂。 想象一下假如一个表单包含一打字段,我们真的将必须去编写每个域对应的if判断语句?
  另外一个问题是表单的重新显示。若数据验证失败后,返回客户端的表单中各字段最好是填有原来提交的数据,以便用户查看哪里出现错误(用户也不需再次填写正确的字段值)。 我们可以手动地将原来的提交数据返回给模板,并且必须编辑HTML里的各字段来填充原来的值。 



1 # views.py
2
3 def contact(request):
4     errors = []
5     if request.method == 'POST':
6         if not request.POST.get('subject', ''):
7             errors.append('Enter a subject.')
8         if not request.POST.get('message', ''):
9             errors.append('Enter a message.')
10         if request.POST.get('email') and '@' not in request.POST['email']:
11             errors.append('Enter a valid e-mail address.')
12         if not errors:
13             send_mail(
14                 request.POST['subject'],
15                 request.POST['message'],
16                 request.POST.get('email', `'noreply@example.com`_'),
17                 [`'siteowner@example.com`_'],
18             )
19             return HttpResponseRedirect('/contact/thanks/')
20     return render_to_response('contact_form.html', {
21         'errors': errors,
22         **'subject': request.POST.get('subject', ''),**
23         **'message': request.POST.get('message', ''),**
24         **'email': request.POST.get('email', ''),**
25     })
26
27 # contact_form.html
28
29
30
31     Contact us
32
33
34     Contact us
35
36     {% if errors %}
37         
38             {% for error in errors %}
39             {{ error }}
40             {% endfor %}
41         
42     {% endif %}
43
44     
45         Subject:
46         Your e-mail (optional):
47         Message: **{{ message }}**
48         
49     
50
51
  在视图中使用Form类
  Django带有一个form库,称为django.forms,这个库可以处理我们本章所提到的包括HTML表单显示以及验证。 接下来我们来深入了解一下form库,并使用它来重写contact表单应用。


  表单框架最主要的用法是,为每一个将要处理的HTML的“"定义一个Form类。 在这个例子中,我们只有一个" ",因此我们只需定义一个Form类。 这个类可以存在于任何地方,甚至直接写在"views.py"文件里也行,但是社区的惯例是把Form类都放到一个文件中:forms.py。在存放" views.py" 的目录中,创建这个文件,然后输入: 


1 from django import forms
2
3 class ContactForm(forms.Form):
4     subject = forms.CharField()
5     email = forms.EmailField(required=False)
6     message = forms.CharField()
  这看上去简单易懂,并且很像在模块中使用的语法。 表单中的每一个字段(域)作为Form类的属性,被展现成Field类。这里只用到CharField和EmailField类型。 每一个字段都默认是必填。要使email成为可选项,我们需要指定required=False。
  接下来,我们在使用它重写contact表单应用: 



1 # views.py
2
3 from django.shortcuts import render_to_response
4 from mysite.contact.forms import ContactForm
5
6 def contact(request):
7     if request.method == 'POST':
8         form = ContactForm(request.POST)
9         if form.is_valid():
10             cd = form.cleaned_data
11             send_mail(
12                 cd['subject'],
13                 cd['message'],
14                 cd.get('email', 'noreply@example.com'),
15                 ['siteowner@example.com'],
16             )
17             return HttpResponseRedirect('/contact/thanks/')
18     else:
19         form = ContactForm()
20     return render_to_response('contact_form.html', {'form': form})
21
22 # contact_form.html
23
24
25
26     Contact us
27
28
29     Contact us
30
31     {% if form.errors %}
32         
33             Please correct the error{{ form.errors|pluralize }} below.
34         
35     {% endif %}
36
37     
38         
39             {{ form.as_table }}
40         
41         
42     
43
44
  
  看看,我们能移除这么多不整齐的代码! Django的forms框架处理HTML显示、数据校验、数据清理和表单错误重现。 
  私人定制Form设计
  修改form的显示的最快捷的方式是使用CSS。 尤其是错误列表,可以增强视觉效果。自动生成的错误列表精确的使用“”,这样,我们就可以针对它们使用CSS。 下面的CSS让错误更加醒目了:





1
2     ul.errorlist {
3         margin: 0;
4         padding: 0;
5     }
6     .errorlist li {
7         background-color: red;
8         color: white;
9         display: block;
10         font-size: 10px;
11         margin: 0 0 3px;
12         padding: 4px 5px;
13     }
14
  虽然,自动生成HTML是很方便的,但是在某些时候,你会想覆盖默认的显示。 {{form.as_table}}和其它的方法在开发的时候是一个快捷的方式,form的显示方式也可以在form中被方便地重写。


  每一个字段部件(, , , 或者类似)都可以通过访问{{form.字段名}}进行单独的渲染。




1
2
3     Contact us
4
5
6     Contact us
7
8     {% if form.errors %}
9         
10             Please correct the error{{ form.errors|pluralize }} below.
11         
12     {% endif %}
13
14     
15         
16             {{ form.subject.errors }}
17             Subject:
18             {{ form.subject }}
19         
20         
21             {{ form.email.errors }}
22             Your e-mail address:
23             {{ form.email }}
24         
25         
26             {{ form.message.errors }}
27             Message:
28             {{ form.message }}
29         
30         
31     
32
33
  {{ form.message.errors }} 会在  里面显示,如果字段是合法的,或者form没有被绑定,就显示一个空字符串。 我们还可以把 form.message.errors 当作一个布尔值或者当它是list在上面做迭代, 例如:



1
2     {% if form.message.errors %}
3         
4         {% for error in form.message.errors %}
5             {{ error }}
6         {% endfor %}
7         
8     {% endif %}
9     Message:
10     {{ form.message }}
11
  在校验失败的情况下, 这段代码会在包含错误字段的div的class属性中增加一个”errors”,在一个有序列表中显示错误信息。
  好了,Django的学习笔记到此就正式结束了,基本的网站搭建管理这点点知识已经足以,恩,基本情况就是酱紫了。
    
    PS:本博客欢迎转发,但请注明博客地址及作者~
    博客地址:http://www.iyunv.com/voidy/
    

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-58916-1-1.html 上篇帖子: python操作MongoDB 下篇帖子: 在.NET外散步之我爱贪吃蛇Python
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表