模板
模板注释
单行注释:
{# 注释内容 #}
data:image/s3,"s3://crabby-images/d05c4/d05c4ab03a59007b5ee3551f61f57afa36defc1d" alt=""
多行注释:
{% comment %}
注释内容
{% endcomment %}
data:image/s3,"s3://crabby-images/6fc6f/6fc6f745d97cf206e7bde2782d83f6a6287fea5d" alt=""
注意,模板注释和html注释不同,html注释后浏览器检查还能看到朱时候的代码,但模板注释不能被看到。
模板的功能
产生html,控制页面上展示的内容。模板文件不仅仅是一个html文件。
模板文件包含两部分内容:
1)静态内容:css,js,html。
2)动态内容:用于动态去产生一些网页内容。通过模板语言来产生。
模板文件的使用
通常是在视图函数中使用模板产生html内容返回给客户端。
a)加载模板文件 loader.get_template
获取模板文件的内容,产生一个模板对象。
b)定义模板上下文 RequeseContext
给模板文件传递数据。
c)模板渲染产生html页面内容 render
用传递的数据替换相应的变量,产生一个替换后的标准的html内容。
不用render自己渲染模板示例/ 给一个函数使用templates模板
def index(request):
# 1.加载模板文件,获取一个模板对象
temp = loader.get_template('booktest/index.html')
# 2.定义模板上下文,给模板文件传数据
# context = RequestContext(request, {}) # 在django1.11中不能使用这种方法,只能直接定义一个字典
context = {}
# 3.模板渲染,产生一个替换后的html内容
res_html = temp.render(context)
# 4.返回应答
return HttpResponse(res_html)
data:image/s3,"s3://crabby-images/734ef/734ef205e8477d4c80f688b948f8f0c82d5f06b9" alt=""
自定义render渲染模板示例/ 抽离出templates渲染函数给多个视图函数使用
def my_render(request, template_path, context={}):
# 1.加载模板文件,获取一个模板对象
temp = loader.get_template(template_path)
# 2.定义模板上下文,给模板文件传数据
# context = RequestContext(request, context)
# context = {}
# 3.模板渲染,产生一个替换后的html内容
res_html = temp.render(context)
# 4.返回应答
return HttpResponse(res_html)
def index(request):
# return render(request, 'booktest/index.html')
return my_render(request, 'booktest/index.html')
data:image/s3,"s3://crabby-images/f392a/f392ae79e564366b80e263807bb564ef1dc817be" alt=""
模板文件加载顺序
1)首先去配置的模板目录下面去找模板文件。
2)去INSTALLED_APPS下面的每个应用的templates去找模板文件,前提是应用中必须有templates文件夹。
# index2页面未创建
def index2(request):
"""模板文件的加载顺序"""
return my_render(request, 'booktest/index2.html')
data:image/s3,"s3://crabby-images/312a8/312a899c5cc4b457588c34b029baf9ef2a679edd" alt=""
报错结果
data:image/s3,"s3://crabby-images/3e49f/3e49fce1b5a694ae591e3c072d54407d80436e3a" alt=""
模板语言
模板语言简称为DTL。(Django Template Language)
模板变量
模板变量名是由数字,字母,下划线和点组成的,不能以下划线开头。
使用模板变量:
<pre>{{模板变量名}}</pre>
模板变量的解析顺序:
例如:{{ book.btitle }}
- 1)首先把book当成一个字典,把btitle当成键名,进行取值book['btitle']
- 2)把book当成一个对象,把btitle当成属性,进行取值book.btitle
- 3)把book当成一个对象,把btitle当成对象的方法,进行取值book.btitle
例如:{{book.0}}
- 1)首先把book当成一个字典,把0当成键名,进行取值book[0]
- 2)把book当成一个列表,把0当成下标,进行取值book[0]
如果解析失败,则产生内容时用空字符串填充模板变量。
使用模板变量时,.前面的可能是一个字典,可能是一个对象,还可能是一个列表。
使用模板变量示例
模板变量函数
def temp_var(request):
"""模板变量"""
my_dict = {"title": "字典键值"}
my_list = [1, 2, 3]
book = models.BookInfo.objects.get(id=1)
# 定义模板上下文
context = {'my_dict': my_dict, 'my_list': my_list, 'book': book}
return render(request, 'booktest/temp_var.html', context)
data:image/s3,"s3://crabby-images/174d7/174d7467fff775bbd629c5c60c3b2f5e4f17dd5d" alt=""
模板变量html代码(booktest/temp_var.html)
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>模板变量</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
</head>
<body>
使用字典属性:{{ my_dict.title }} <br>
使用列表元素:{{ my_list.1 }} <br>
使用对象属性:{{ book.btitle }}
</body>
</html>
data:image/s3,"s3://crabby-images/ddd42/ddd42fd60607cd0e72a74811fbf2db62102ef30e" alt=""
模板标签
模板标签使用格式:
{% 代码段 %}
data:image/s3,"s3://crabby-images/5b81f/5b81fd0c4997c8455e059668699e67e9d3a7887a" alt=""
for循环:
{% for x in 列表 %}
# 列表不为空时执行
{% empty %}
# 列表为空时执行
{% endfor %}
data:image/s3,"s3://crabby-images/f1be8/f1be84b2e3758cebdafd2e47665c9b0cb3e234f5" alt=""
可以通过{{ forloop.counter }}得到for循环遍历到了第几次。
if判断:
{% if 条件 %}
{% elif 条件 %}
{% else %}
{% endif %}
data:image/s3,"s3://crabby-images/fc0ba/fc0bac31e6cff3be953eebcda73d941d42feebf9" alt=""
关系比较操作符:> < >= <= == !=
注意:进行比较操作时,比较操作符两边必须有空格。
逻辑运算:not and or
django内置标签与过滤器文档:内置标签与过滤器文档 或者 https://yiyibooks.cn/xx/django_182/ref/templates/builtins.html
模板标签使用示例
函数
def temp_tags(request):
"""模板标签"""
books = models.BookInfo.objects.all()
return render(request, 'booktest/temp_tags.html', {'books': books})
data:image/s3,"s3://crabby-images/d0579/d057946a68a2033d59eee305990679ad6a6fa522" alt=""
html代码
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>模板标签</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
<style>
.red {background-color: red;}
.blue {background-color: blue;}
.gray {background-color: gray;}
</style>
</head>
<body>
<h1>模板标签</h1>
<ul>
{% for book in books %}
{% if book.id <= 2 %}
<li class="red">{{ forloop.counter }}--{{ book.btitle }}</li>
{% elif book.id <= 3 %}
<li class="blue">{{ forloop.counter }}--{{ book.btitle }}</li>
{% else %}
<li class="gray">{{ forloop.counter }}--{{ book.btitle }}</li>
{% endif %}
{% endfor %}
</ul>
</body>
</html>
booktest/temp_tags.html
data:image/s3,"s3://crabby-images/0872a/0872a04400249ea2fa64f048a67e8ca437d24baa" alt=""
过滤器
过滤器其实就是python函数。
过滤器用于对模板变量进行操作。
- date:改变日期的显示格式。
- length:求长度。字符串,列表.
- default:设置模板变量的默认值。
格式:
模板变量|过滤器:参数
data:image/s3,"s3://crabby-images/b2389/b23892d5c53acf926d739ab9b3ee7c33169db2d5" alt=""
自定义过滤器。
自定义的过滤器函数,至少有一个参数,最多两个
过滤器的使用示例
过滤器函数
def temp_filter(request):
"""模板过滤器"""
books = models.BookInfo.objects.all()
return render(request, 'booktest/temp_filter.html', {'books': books})
data:image/s3,"s3://crabby-images/20ccd/20ccdf14f1054fc1b59a00c8fe746a89c6072965" alt=""
过滤器html代码
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>模板过滤器</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
<style>
.red {background-color: red;}
.blue {background-color: blue;}
.gray {background-color: gray;}
</style>
</head>
<body>
<h1>模板标签</h1>
<ul>
{% for book in books %}
{% if book.id <= 2 %}
<li class="red">{{ book.btitle }}--{{ book.bpub_date|date:'Y-m-d' }}</li>
{% else %}
{# 加了length后,会将原来的图书名编程图书名长度的数字 #}
<li class="gray">{{ book.btitle|length }}--{{ book.bpub_date }}</li>
{% endif %}
{% endfor %}
</ul>
default过滤器:
{# 如果content没有值,则过滤器会显示默认的值 #}
{{ content|default:"hello" }}
</body>
</html>
booktest/temp_filter.html
data:image/s3,"s3://crabby-images/206f3/206f3fbd4645aa109f9d39fa0d21a4c967314a9f" alt=""
注意:过滤器中,冒号后不能加空格,
例如{{ book.bpub_date|date:'Y-m-d' }}是正确的,而{{ book.bpub_date|date: 'Y-m-d' }}是错误的。
否则会报错,Could not parse the remainder: ': 'Y-m-d'' from 'book.bpub_date|date: 'Y-m-d''
自定义过滤器
自定义过滤器
- 1.在自己的应用下面新建一个 templatetags 文件夹,名字固定;
- 2.在templatetags文件夹下面新建一个py文件,名字自定义,比如filters.py;
- 3.1.在文件中,引入Library类;
- 3.2.创建一个Library类的对象;
- 3.3.定义自己的函数,给函数添加装饰器@register.filter进行过滤器装饰;
使用自定义装饰器
- 1.在需要使用的html文件中导入自定义装饰器文件,{% load filters %},即 load 文件名;
- 2.使用过滤器;
自定义过滤器示例
import django.template
# 创建一个Library类的对象
register = django.template.Library()
@register.filter
def mod(num):
"""判断num是否为偶数"""
# 如果传过来的num是偶数,则返回True,否则返回False
return num % 2 == 0
data:image/s3,"s3://crabby-images/2fb91/2fb912fcc4b61d7b2fed985f05cc2d6fb402ccad" alt=""
在html代码中使用模板过滤器
<!DOCTYPE html>
<html lang="en">
{% load filters %}
<head>
<meta charset="UTF-8">
<title>模板过滤器</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
<style>
.red {background-color: red;}
.blue {background-color: blue;}
.gray {background-color: gray;}
</style>
</head>
<body>
<h1>模板标签</h1>
<ul>
{% for book in books %}
{% if book.id|mod %}
<li class="red">{{ book.id }}--{{ book.btitle }}--{{ book.bpub_date|date:'Y-m-d' }}</li>
{% else %}
<li class="gray">{{ book.btitle }}--{{ book.bpub_date }}</li>
{% endif %}
{% endfor %}
</ul>
</body>
</html>
html使用一个参数的过滤器
data:image/s3,"s3://crabby-images/834fb/834fb62c1c8450ff2957acc9a8a2b04ff7ddb220" alt=""
两个参数的自定义过滤器
自定义过滤器
# 自定义的过滤器,最少有一个参数,最多有两个参数
# 只有一个参数的话,由|前面的参数传过来,两个参数的话,:后面再跟一个参数
# 应注意过滤器参数的前后顺序
@register.filter
def mod_val(num, val):
"""判断num是否能把value整除"""
return num % val == 0
data:image/s3,"s3://crabby-images/9bd94/9bd9484011713be1ebc12fa96b18918dd518c703" alt=""
html代码
<!DOCTYPE html>
<html lang="en">
{% load filters %}
<head>
<meta charset="UTF-8">
<title>模板过滤器</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
<style>
.red {background-color: red;}
.blue {background-color: blue;}
.gray {background-color: gray;}
</style>
</head>
<body>
<h1>模板标签</h1>
<ul>
{% for book in books %}
{# {% if book.id <= 2 %} #}
{# % if book.id|mod %} #}
{% if book.id|mod_val:3 %}
<li class="red">{{ book.id }}--{{ book.btitle }}--{{ book.bpub_date|date:'Y-m-d' }}</li>
{% else %}
{# 加了length后,会将原来的图书名编程图书名长度的数字 #}
<li class="gray">{{ book.btitle|length }}--{{ book.bpub_date }}</li>
{% endif %}
{% endfor %}
</ul>
default过滤器:
{# 如果content没有值,则过滤器会显示默认的值 #}
{{ content|default:"hello" }}
</body>
</html>
html使用两个参数的自定义过滤器
data:image/s3,"s3://crabby-images/53715/53715fd0223c9d05aa59b0a54389972559c1471e" alt=""
模板继承
模板继承也是为了重用html页面内容。
data:image/s3,"s3://crabby-images/81231/8123147650e8bccd2f3172b2ba036e4a49ca63ef" alt=""
在父模板里可以定义块,使用标签:
<pre>{% block 块名 %}
块中间可以写内容,也可以不写
{% endblock 块名%}</pre>
子模板去继承父模板之后,可以重写父模板中的某一块的内容。
继承格式:
{% extends 父模板文件路径%}
{% block 块名 %}
{{ block.super}} #获取父模板中块的默认内容
重写的内容
{% endblock 块名%}
data:image/s3,"s3://crabby-images/40776/4077672d233e586f167bc31bd69ce02e3d25e119" alt=""
模板继承示例
base/母模板
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>{% block title %}base模板文件的title{% endblock title %}</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
</head>
<body>
<h1>base模板的header</h1>
{% block b1 %}
<h2>base模板的block1</h2>
{% endblock b1 %}
{% block b2 %}
<h2>base模板的block1</h2>
{% endblock b2 %}
<h1>base模板的footer</h1>
</body>
</html>
data:image/s3,"s3://crabby-images/01c64/01c645c29fc321976ce59bc3b2efe7d094700f92" alt=""
child/子模板
{% extends 'booktest/base.html' %}
{% block title %}child模板中的title{% endblock title %}
{% block b1 %}
{{ block.super }}
<h2>child模板的block1</h2>
{% endblock b1 %}
{% block b2 %}
<h2>child模板的block2</h2>
{% endblock b2 %}
data:image/s3,"s3://crabby-images/b3c87/b3c87477932e1b3badb0c92d96c00b29f9892102" alt=""
配套函数
def temp_inherit(request):
"""返回child模板继承于base的文件"""
return render(request, 'booktest/child.html')
data:image/s3,"s3://crabby-images/fa371/fa3710f503a7b8590c8c0102aa56078a3c483de7" alt=""
在块里面还可以写块,需要注意endblock后面跟 块名 用以区分哪个块结束。
html转义
场景:编辑商品详情信息,数据表中保存的是html内容。
在模板上下文中的html标记默认是会被转义的。
小于号< 转换为<
大于号> 转换为>
单引号' 转换为'
双引号" 转换为 "
与符号& 转换为 &
data:image/s3,"s3://crabby-images/a3740/a3740b532071f079a69527cb5f0a48c96dc44238" alt=""
要关闭模板上下文字符串的转义:可以使用
{{ 模板变量|safe}}
data:image/s3,"s3://crabby-images/aee80/aee80fdd2c4ce65ae58b1ef07c42e846bcab0c59" alt=""
也可以使用:
{% autoescape off %}
模板语言代码
{% endautoescape %}
data:image/s3,"s3://crabby-images/f89a5/f89a5293dafdb328e1389d18e273a014c07de904" alt=""
模板硬编码中的字符串默认不会经过转义,如果需要转义,那需要手动进行转义。
safe和autoescape的区别在于,safe只能转义一个模板变量,而autoescape可以转义多个变量;
html转义示例
html_escape.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>模板转义</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
</head>
<body>
html转义结果:<br>
{{ content }}<br>
使用safe过滤器关闭转义结果:<br>
{{ content|safe }}<br>
使用autoescape关闭转义结果:<br>
{% autoescape off %}
{{ content }}
{{ content }}
{% endautoescape %}<br>
模板硬编码中的字符串默认不会转义,即会经过html渲染:<br>
{{ test|default:'<h1>hello</h1>' }}<br>
手动对硬编码进行转义:<br>
{{ test|default:'<h1>hello</h1>' }}
</body>
</html>
data:image/s3,"s3://crabby-images/d40cc/d40cca32f318ac5f9b74f58304fec42b93d8d7de" alt=""
配套函数
# /html_escape
def html_escape(request):
"""模板转义"""
return render(request, 'booktest/html_escape.html', {'content': '<h1>hello</h1>'})
data:image/s3,"s3://crabby-images/55c4e/55c4e115226b22b1aaa664e4645670b9cf97a19d" alt=""
验证码
在用户注册、登录页面,为了防止暴力请求,可以加入验证码功能,
如果验证码错误,则不需要继续处理,可以减轻业务服务器、数据库服务器的压力。
自实现验证码
安装pillow
pip install Pillow==3.4.1
data:image/s3,"s3://crabby-images/2aae4/2aae42bc21a1e19b7855164fb0aea146ceb77faa" alt=""
编写函数/视图
from PIL import Image, ImageDraw, ImageFont
from django.utils.six import BytesIO
def verify_code(request):
# 引入随机函数模块
import random
# 定义变量,用于画面的背景色、宽、高
bgcolor = (random.randrange(20, 100), random.randrange(
20, 100), 255)
width = 100
height = 25
# 创建画面对象
im = Image.new('RGB', (width, height), bgcolor)
# 创建画笔对象
draw = ImageDraw.Draw(im)
# 调用画笔的point()函数绘制噪点
for i in range(0, 100):
xy = (random.randrange(0, width), random.randrange(0, height))
fill = (random.randrange(0, 255), 255, random.randrange(0, 255))
draw.point(xy, fill=fill)
# 定义验证码的备选值
str1 = 'ABCD123EFGHIJK456LMNOPQRS789TUVWXYZ0'
# 随机选取4个值作为验证码
rand_str = ''
for i in range(0, 4):
rand_str += str1[random.randrange(0, len(str1))]
# 构造字体对象,ubuntu的字体路径为“/usr/share/fonts/truetype/freefont”
font = ImageFont.truetype('FreeMono.ttf', 23)
# 构造字体颜色
fontcolor = (255, random.randrange(0, 255), random.randrange(0, 255))
# 绘制4个字
draw.text((5, 2), rand_str[0], font=font, fill=fontcolor)
draw.text((25, 2), rand_str[1], font=font, fill=fontcolor)
draw.text((50, 2), rand_str[2], font=font, fill=fontcolor)
draw.text((75, 2), rand_str[3], font=font, fill=fontcolor)
# 释放画笔
del draw
# 存入session,用于做进一步验证
request.session['verifycode'] = rand_str
# 内存文件操作
buf = BytesIO()
# 将图片保存在内存中,文件类型为png
im.save(buf, 'png')
# 将内存中的图片数据返回给客户端,MIME类型为图片png
return HttpResponse(buf.getvalue(), 'image/png')
自实现验证码函数
data:image/s3,"s3://crabby-images/81d70/81d707b8b7fdf552a41f2e9ea2da39dca230c85f" alt=""
查看验证码
url(r'^verify_code/$', views.verify_code),
data:image/s3,"s3://crabby-images/d20e6/d20e6f23b0327b6ff66a02e6810d9039fecfb680" alt=""
在网页中使用验证码
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Login</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
</head>
<body>
<form action="/login_check/" method="post">
{% csrf_token %}
用户:<input type="text" id="username" name="username" value="{{ usernam }}"><br />
密码:<input type="password" id="password" name="password"><br />
<input type="checkbox" name="remember">记住用户名<br />
<img src="/verify_code"> <input type="text" name="vcode"><br>
<input type="submit" value="登录">
</form>
</body>
</html>
html中使用验证码
data:image/s3,"s3://crabby-images/dcc37/dcc37029c809db69ed55b01d772de592c9594469" alt=""
登录校验验证码函数
def login_check(request):
# 1.获取用户名和密码
username = request.POST.get("username")
password = request.POST.get("password")
remember = request.POST.get("remember")
print(username, password)
# 获取用户输入的验证码
vcode1 = request.POST.get('vcode')
# 获取session中保存的验证码
vcode2 = request.session.get('verifycode')
# 进行验证码校验
if vcode1 != vcode2:
# 验证码错误
return redirect('/login')
# 2.进行校验
# 3.返回应答
if username == "yifchan" and password == "yifchan":
# response = redirect("/index")
response = redirect("/change_pwd")
# 判断是否需要记住用户名
if remember == 'on':
# 设置cookie username-过期时间为1周
response.set_cookie('username', username, max_age=7 * 24 * 3600)
# 记住用户登录状态
# 只有session中有islogin,就认为用户已经登录
request.session['islogin'] = True
request.session['username'] = username
return response
else:
return HttpResponse("账号或密码错误")
验证码校验函数
data:image/s3,"s3://crabby-images/f334c/f334ce1e74ae4fd0331239bae7d05448b9254d46" alt=""
url反向解析
当某一个url配置的地址发生变化时,页面上使用反向解析生成地址的位置不需要发生变化。
根据url 正则表达式的配置动态的生成url。
在项目urls中包含具体应用的urls文件时指定namespace;
urlpatterns = [
url(r'^admin/', admin.site.urls),
url(r'^', include('booktest.urls', namespace='booktest')),
]
data:image/s3,"s3://crabby-images/32d65/32d65390a6695e3abb313ac443bafc0bc348642e" alt=""
在应用的urls中配置是指定name;
urlpatterns = [
url(r'^index/$', views.index, name='index'),
url(r'^url_reverse/$', views.url_reverse), #
url(r'^show_args/(\d+)/(\d+)$', views.show_args, name='show_args'), # 捕获位置参数
url(r'^show_kwargs/(?P<c>\d+)/(?P<d>\d+)$', views.show_kwargs, name='show_kwargs'), # 捕获关键字参数
]
data:image/s3,"s3://crabby-images/05b69/05b69483fefc60606fb63a54ef9c56fcd7c56b35" alt=""
在模板文件中使用时,格式如下:
{% url 'namespace名字:name' %} 例如{% url 'booktest:fan2' %}
data:image/s3,"s3://crabby-images/dea1f/dea1f9deb6965258949eb7bb07a7a9871b59578e" alt=""
带位置参数:
{% url 'namespace名字:name' 参数 %} 例如{% url 'booktest:fan2' 1 %}
data:image/s3,"s3://crabby-images/200a3/200a3a4685d5d02772afc4fdbafd771428507666" alt=""
带关键字参数:
{% url 'namespace名字:name' 关键字参数 %} 例如{% url 'booktest:fan2' id=1 %}
data:image/s3,"s3://crabby-images/714b4/714b45448f09602be3818b5a8d773e8c4878fd8a" alt=""
在视图中/重定向的时候使用反向解析:
from django.core.urlresolvers import reverse
data:image/s3,"s3://crabby-images/fd102/fd102904d9c72ed3d4f6412b599d3736541637e1" alt=""
无参数:
reverse('namespace名字:name名字')
data:image/s3,"s3://crabby-images/0c790/0c7902613fb550939bce967992fb13da85ecf4d3" alt=""
如果有位置参数
reverse('namespace名字:name名字', args = 位置参数元组)
data:image/s3,"s3://crabby-images/26e7b/26e7bcd9b2b608a334bfdb7b468908bfb15eac32" alt=""
如果有关键字参数
reverse('namespace名字:name名字', kwargs=字典)
data:image/s3,"s3://crabby-images/43bfc/43bfc4d69d6340e916d5f12165b5cf08fd86664a" alt=""
url反向解析示例
项目下的url文件
urlpatterns = [
url(r'^admin/', admin.site.urls),
url(r'^', include('booktest.urls', namespace='booktest')),
]
data:image/s3,"s3://crabby-images/650db/650dbca0e801a8fa80512f3f918a8d4d1580f449" alt=""
应用下的url文件
urlpatterns = [
url(r'^index/$', views.index, name='index'),
url(r'^url_reverse/$', views.url_reverse), #
url(r'^show_args/(\d+)/(\d+)$', views.show_args, name='show_args'), # 捕获位置参数
url(r'^show_kwargs/(?P<c>\d+)/(?P<d>\d+)$', views.show_kwargs, name='show_kwargs'), # 捕获关键字参数
url(r'^test_reverse/$', views.test_reverse, name='test_reverse'), # 视图使用反向解析
]
data:image/s3,"s3://crabby-images/51dc6/51dc6380d1dfa33747ea07ea4661427983fab6e1" alt=""
模板中使用url反向解析
配套函数
def url_reverse(request):
return render(request, 'booktest/url_reverse.html')
def show_args(request, a, b):
return HttpResponse(a+':'+b)
def show_kwargs(request, c, d):
return HttpResponse(c+':'+d)
data:image/s3,"s3://crabby-images/5c876/5c876bcc40827a55f0c4aecbebed280eae181bef" alt=""
url_reverse.html
data:image/s3,"s3://crabby-images/38282/382827d9b282b4f4fbc7ecb129d9f221e6998df6" alt=""
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>url反向解析</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
</head>
<body>
index链接:<br>
<a href="/index">首页</a><br>
url反向解析生成index链接:<br>
<a href="{% url 'booktest:index' %}">首页</a><br>
位置参数 /show_args/1/2:<br>
<a href="/show_args/1/2">/show_args/1/2</a><br>
动态生成/show_args/1/2:<br>
<a href="{% url 'booktest:show_args' 1 2 %}">/show_args/1/2</a><br>
关键字参数 /show_kwargs/3/4:<br>
<a href="/show_kwargs/3/4">/show_args/3/4</a><br>
动态生成/show_kwargs/3/4:<br>
<a href="{% url 'booktest:show_kwargs' c=3 d=4 %}">/show_args/3/4</a><br>
</body>
</html>
data:image/s3,"s3://crabby-images/3bc23/3bc23fa2ebb6ad2560136048121a6668721ea6b1" alt=""
视图中使用反向解析
反向解析函数
from django.core.urlresolvers import reverse
# url: /test_reverse
def test_reverse(request):
# 重定向到 /index
# return redirect('/index')
# url = reverse('booktest:index')
# 重定向到位置参数 show_args/1/2
# url = reverse('booktest:show_args', args=(1, 2))
# 重定向到关键字参数 show_kwargs/3/4
url = reverse('booktest:show_kwargs', kwargs={'c': 3, 'd': 4})
return redirect(url)
data:image/s3,"s3://crabby-images/fe3c0/fe3c06e8215db5045ceb009eb94d5285a30aabc9" alt=""
csrf攻击
首先做一个登录页,让用户输入用户名和密码进行登录,登录成功之后跳转的修改密码页面。在修改密码页面输入新密码,点击确认按钮完成密码修改。
登录页需要一个模板文件login.html.修改密码页面也需要一个模板文件change_pwd.html.
显示登录页的视图login,验证登录的视图login_check,显示发帖页的视图change_pwd,处理修改密码的视图change_pwd_action.
加功能:
a)只有用户登录之后才可以进行修改密码操作。
登录装饰器函数。
def login_required(view_func):
"""登录判断装饰器"""
def wrapper(request, *args, **kwargs):
# 判断用户是否登录
if request.session.has_key('islogin'):
# 已经登录了,就返回到函数
return view_func(request, *args, **kwargs)
else:
# 未登录,返回到登录页面
return redirect('/login')
return wrapper
data:image/s3,"s3://crabby-images/34bf5/34bf54a632cabe55609cdc771a7835f9cfeba007" alt=""
案例流程图:
data:image/s3,"s3://crabby-images/53b09/53b093e36c3959524519af053b519710f252158f" alt=""
django防止csrf的方式:
- 默认打开csrf中间件。
- 表单post提交数据时加上{% csrf_token %}标签。
防御原理:
- 1)渲染模板文件时在页面生成一个名字叫做csrfmiddlewaretoken的隐藏域。
- 2)服务器交给浏览器保存一个名字为csrftoken的cookie信息。
- 3)提交表单时,两个值都会发给服务器,服务器进行比对,如果一样,则csrf验证通过,否则失败。
登录案例代码
设计路由
url(r'^login/$', views.login), # 返回登录页面
url(r'^login_check/$', views.login_check), # 进行登录校验
url(r'^change_pwd/$', views.change_pwd), # 返回修改密码页面
url(r'^change_pwd_action/$', views.change_pwd_action), # 修改密码处理函数
data:image/s3,"s3://crabby-images/997eb/997eb78c9753eab154c9c10836606c6d82db4976" alt=""
编写login.html页面
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Login</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
</head>
<body>
<form action="/login_check/" method="post">
<label for="username">用户:
<input type="text" id="username" name="username" value="{{ usernam }}">
</label><br />
<label for="password">密码:
<input type="password" id="password" name="password">
</label><br />
<input type="checkbox" name="remember">记住用户名<br />
<input type="submit" value="登录">
</form>
</body>
</html>
login.html
data:image/s3,"s3://crabby-images/5ffdf/5ffdfd572e30014f582cc9742eb1ff3bc9b0e8db" alt=""
编写login函数
def login(request):
"""登录处理函数"""
# 判断用户是否登录
if request.session.has_key('islogin'):
# return redirect('/index')
return redirect('/change_pwd')
else:
# 获取cookie username
if 'username' in request.COOKIES:
username = request.COOKIES['username']
else:
username = ''
return render(request, "booktest/login.html", {'usernam': username})
login函数
data:image/s3,"s3://crabby-images/f757a/f757ad429770b3ca189e765c03de8e57944cc749" alt=""
编写
def login_check(request):
# 1.获取用户名和密码
username = request.POST.get("username")
password = request.POST.get("password")
remember = request.POST.get("remember")
print(username, password)
# 2.进行校验
# 3.返回应答
if username == "yifchan" and password == "yifchan":
# response = redirect("/index")
response = redirect("/change_pwd")
# 判断是否需要记住用户名
if remember == 'on':
# 设置cookie username-过期时间为1周
response.set_cookie('username', username, max_age=7*24*3600)
# 记住用户登录状态
# 只有session中有islogin,就认为用户已经登录
request.session['islogin'] = True
request.session['username'] = username
return response
else:
return HttpResponse("账号或密码错误")
login_check函数
data:image/s3,"s3://crabby-images/bcdc0/bcdc0a59d97e24759ea7207572cf5aaaae4711a9" alt=""
编写change_pwd.html页面
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>修改密码</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
</head>
<body>
<form action="/change_pwd_action">
新密码:<input type="password" name="pwd">
<input type="submit" value="确认修改">
</form>
</body>
</html>
change_pwd.html
data:image/s3,"s3://crabby-images/8c5e0/8c5e00a220eec189d4fe56d482ee0ec40f81a2aa" alt=""
编写登录判断装饰器
def login_required(view_func):
"""登录判断装饰器"""
def wrapper(request, *args, **kwargs):
# 判断用户是否登录
if request.session.has_key('islogin'):
# 已经登录了,就返回到函数
return view_func(request, *args, **kwargs)
else:
# 未登录,返回到登录页面
return redirect('/login')
return wrapper
登录判断装饰器
data:image/s3,"s3://crabby-images/a6a71/a6a7146dfa930fa4d7c710f97cdeda42a08a5d7a" alt=""
定义change_pwd函数
@login_required
def change_pwd(request):
"""显示修改密码页面"""
return render(request, 'booktest/change_pwd.html')
data:image/s3,"s3://crabby-images/0fb3b/0fb3bfc887057ff9499272a055db80e41d6f15fa" alt=""
定义change_pwd_action函数
@login_required
def change_pwd_action(request):
"""模拟修改密码处理"""
# 1.获取新密码
pwd = request.POST.get("pwd")
# username = request.COOKIES['username']
username = request.session.get('username')
# 2.实际开发的时候,修改对应的数据库中的数据
# 3.返回应答
return HttpResponse('%s修改密码为:%s' % (username, pwd))
change_pwd_action函数
data:image/s3,"s3://crabby-images/5a23f/5a23f7e3c83857d3a869a7ba0a366050b083d47a" alt=""
csrf伪造
django默认开启csrf防护,并且只针对post提交;
- 1.开启csrf防护,把注释的settings文件里面的csrf放开注释;
- 2.在要提交表单的地方,添加{% csrf_token %}
csrf防护示例代码
login页面
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Login</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
</head>
<body>
<form action="/login_check/" method="post">
{% csrf_token %}
用户:<input type="text" id="username" name="username" value="{{ usernam }}"><br />
密码:<input type="password" id="password" name="password"><br />
<input type="checkbox" name="remember">记住用户名<br />
<input type="submit" value="登录">
</form>
</body>
</html>
login.html
data:image/s3,"s3://crabby-images/44262/442622a1ff364aafe9e1d2145b47c940c72dfb22" alt=""
change_pwd页面
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>修改密码</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
</head>
<body>
<form action="/change_pwd_action/" method="post">
{% csrf_token %}
新密码:<input type="password" name="pwd">
<input type="submit" value="确认修改">
</form>
</body>
</html>
change_pwd.html
data:image/s3,"s3://crabby-images/e087d/e087d766519a767e0eea19b90965463d48b075a4" alt=""
在模拟的时候,自制的第三方网站,可以拿到自己页面的csrftoken隐藏域的信息,然后放到自己的第三方网站代码去;
但在实际上,这样很难实现。
网友评论