PIL简介
什么是PIL
PIL:是Python Image Library的缩写,图像处理的模块。主要的类包括Image,ImageFont,ImageDraw,ImageFilterhtml
PIL的导入
首先须要安装一下pillow包python
1
|
pip install pillow
|
而后就能够调用PIL里的类了django
1
2
3
4
|
from
PIL
import
Image
from
PIL
import
ImageFont
from
PIL
import
ImageDraw
from
PIL
import
ImageFilter
|
PIL经常使用方法
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
|
open
()
#打开图片
new(mode,size,color)
#建立一张空白图片
save(
"test.gif"
,
"GIF"
)
#保存(新图片路径和名称,保存格式)
size()
#获取图片大小
thumbnail(weight,high)
#缩放图片大小(宽,高)
show()
#显示图片
blend(img1,img2,alpha)
#两张图片相加,alpha表示img1和img2的比例参数。
crop()
#剪切,提取某个矩阵大小的图像。它接收一个四元素的元组做为参数,各元素为(left, upper, right, lower),坐标系统的原点(0, 0)是左上角。
rotate(
45
)
#逆时针旋转45度
transpose()
#旋转图像
transpose(Image.FLIP_LEFT_RIGHT)
#左右对换。
transpose(Image.FLIP_TOP_BOTTOM)
#上下对换。
transpose(Image.ROTATE_90)
#旋转 90 度角。
transpose(Image.ROTATE_180)
#旋转 180 度角。
transpose(Image.ROTATE_270)
#旋转 270 度角。
paste(im,box)
#粘贴box大小的im到原先的图片对象中。
convert()
#用来将图像转换为不一样色彩模式。
filters()
#滤镜
BLUR
#虚化
EMBOSS
resize((
128
,
128
))
#resize成128*128像素大小
convert(
"RGBA"
)
#图形类型转换
getpixel((
4
,
4
))
#获取某个像素位置的值
putpixel((
4
,
4
),(
255
,
0
,
0
))
#写入某个像素位置的值
|
PIL应用
咱们主要用PIL来生成一张验证码的随机图,下面,咱们就一步步来作一个小示例session
一、生成一张固定尺寸固定颜色的图片
1
2
3
4
5
6
|
from
PIL
import
Image
# 获取一个Image对象,参数分别是RGB模式。宽150,高30,红色
image
=
Image.new(
'RGB'
,(
150
,
30
),
'red'
)
# 保存到硬盘,名为test.png格式为png的图片
image.save(
open
(
'test.png'
,
'wb'
),
'png'
)
|
二、生成一张随机颜色的图片
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
from
PIL
import
Image
import
random
def
getRandomColor():
'''获取一个随机颜色(r,g,b)格式的'''
c1
=
random.randint(
0
,
255
)
c2
=
random.randint(
0
,
255
)
c3
=
random.randint(
0
,
255
)
return
(c1,c2,c3)
# 获取一个Image对象,参数分别是RGB模式。宽150,高30,随机颜色
image
=
Image.new(
'RGB'
,(
150
,
30
),getRandomColor())
# 保存到硬盘,名为test.png格式为png的图片
image.save(
open
(
'test.png'
,
'wb'
),
'png'
)
|
三、生成一张带有固定字符串的随机颜色的图片
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
|
from
PIL
import
Image
from
PIL
import
ImageDraw
from
PIL
import
ImageFont
import
random
def
getRandomColor():
'''获取一个随机颜色(r,g,b)格式的'''
c1
=
random.randint(
0
,
255
)
c2
=
random.randint(
0
,
255
)
c3
=
random.randint(
0
,
255
)
return
(c1,c2,c3)
# 获取一个Image对象,参数分别是RGB模式。宽150,高30,随机颜色
image
=
Image.new(
'RGB'
,(
150
,
30
),getRandomColor())
# 获取一个画笔对象,将图片对象传过去
draw
=
ImageDraw.Draw(image)
# 获取一个font字体对象参数是ttf的字体文件的目录,以及字体的大小
font
=
ImageFont.truetype(
"kumo.ttf"
,size
=
32
)
# 在图片上写东西,参数是:定位,字符串,颜色,字体
draw.text((
20
,
0
),
'fuyong'
,getRandomColor(),font
=
font)
# 保存到硬盘,名为test.png格式为png的图片
image.save(
open
(
'test.png'
,
'wb'
),
'png'
)
|
效果:app
四、生成一张带有随机字符串随机颜色的图片
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
|
from
PIL
import
Image
from
PIL
import
ImageDraw
from
PIL
import
ImageFont
import
random
def
getRandomColor():
'''获取一个随机颜色(r,g,b)格式的'''
c1
=
random.randint(
0
,
255
)
c2
=
random.randint(
0
,
255
)
c3
=
random.randint(
0
,
255
)
return
(c1,c2,c3)
def
getRandomStr():
'''获取一个随机字符串,每一个字符的颜色也是随机的'''
random_num
=
str
(random.randint(
0
,
9
))
random_low_alpha
=
chr
(random.randint(
97
,
122
))
random_upper_alpha
=
chr
(random.randint(
65
,
90
))
random_char
=
random.choice([random_num, random_low_alpha, random_upper_alpha])
return
random_char
# 获取一个Image对象,参数分别是RGB模式。宽150,高30,随机颜色
image
=
Image.new(
'RGB'
,(
150
,
30
),getRandomColor())
# 获取一个画笔对象,将图片对象传过去
draw
=
ImageDraw.Draw(image)
# 获取一个font字体对象参数是ttf的字体文件的目录,以及字体的大小
font
=
ImageFont.truetype(
"kumo.ttf"
,size
=
26
)
for
i
in
range
(
5
):
# 循环5次,获取5个随机字符串
random_char
=
getRandomStr()
# 在图片上一次写入获得的随机字符串,参数是:定位,字符串,颜色,字体
draw.text((
10
+
i
*
30
,
0
),random_char , getRandomColor(), font
=
font)
# 保存到硬盘,名为test.png格式为png的图片
image.save(
open
(
'test.png'
,
'wb'
),
'png'
)
|
效果:dom
五、生成一张带有噪点的验证码图片
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
|
from
PIL
import
Image
from
PIL
import
ImageDraw
from
PIL
import
ImageFont
import
random
def
getRandomColor():
'''获取一个随机颜色(r,g,b)格式的'''
c1
=
random.randint(
0
,
255
)
c2
=
random.randint(
0
,
255
)
c3
=
random.randint(
0
,
255
)
return
(c1,c2,c3)
def
getRandomStr():
'''获取一个随机字符串,每一个字符的颜色也是随机的'''
random_num
=
str
(random.randint(
0
,
9
))
random_low_alpha
=
chr
(random.randint(
97
,
122
))
random_upper_alpha
=
chr
(random.randint(
65
,
90
))
random_char
=
random.choice([random_num, random_low_alpha, random_upper_alpha])
return
random_char
# 获取一个Image对象,参数分别是RGB模式。宽150,高30,随机颜色
image
=
Image.new(
'RGB'
,(
150
,
30
),getRandomColor())
# 获取一个画笔对象,将图片对象传过去
draw
=
ImageDraw.Draw(image)
# 获取一个font字体对象参数是ttf的字体文件的目录,以及字体的大小
font
=
ImageFont.truetype(
"kumo.ttf"
,size
=
26
)
for
i
in
range
(
5
):
# 循环5次,获取5个随机字符串
random_char
=
getRandomStr()
# 在图片上一次写入获得的随机字符串,参数是:定位,字符串,颜色,字体
draw.text((
10
+
i
*
30
,
0
),random_char , getRandomColor(), font
=
font)
# 噪点噪线
width
=
150
height
=
30
# 划线
for
i
in
range
(
5
):
x1
=
random.randint(
0
,width)
x2
=
random.randint(
0
,width)
y1
=
random.randint(
0
,height)
y2
=
random.randint(
0
,height)
draw.line((x1,y1,x2,y2),fill
=
getRandomColor())
# 画点
for
i
in
range
(
30
):
draw.point([random.randint(
0
, width), random.randint(
0
, height)], fill
=
getRandomColor())
x
=
random.randint(
0
, width)
y
=
random.randint(
0
, height)
draw.arc((x, y, x
+
4
, y
+
4
),
0
,
90
, fill
=
getRandomColor())<br>
# 保存到硬盘,名为test.png格式为png的图片
image.save(
open
(
'test.png'
,
'wb'
),
'png'
)
|
效果:post
六、对验证码图片生成进行封装
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
|
from
PIL
import
Image
from
PIL
import
ImageDraw
from
PIL
import
ImageFont
import
random
class
ValidCodeImg:
def
__init__(
self
,width
=
150
,height
=
30
,code_count
=
5
,font_size
=
32
,point_count
=
20
,line_count
=
3
,img_format
=
'png'
):
'''
能够生成一个通过降噪后的随机验证码的图片
:param width: 图片宽度 单位px
:param height: 图片高度 单位px
:param code_count: 验证码个数
:param font_size: 字体大小
:param point_count: 噪点个数
:param line_count: 划线个数
:param img_format: 图片格式
:return 生成的图片的bytes类型的data
'''
self
.width
=
width
self
.height
=
height
self
.code_count
=
code_count
self
.font_size
=
font_size
self
.point_count
=
point_count
self
.line_count
=
line_count
self
.img_format
=
img_format
@staticmethod
def
getRandomColor():
'''获取一个随机颜色(r,g,b)格式的'''
c1
=
random.randint(
0
,
255
)
c2
=
random.randint(
0
,
255
)
c3
=
random.randint(
0
,
255
)
return
(c1,c2,c3)
@staticmethod
def
getRandomStr():
'''获取一个随机字符串,每一个字符的颜色也是随机的'''
random_num
=
str
(random.randint(
0
,
9
))
random_low_alpha
=
chr
(random.randint(
97
,
122
))
random_upper_alpha
=
chr
(random.randint(
65
,
90
))
random_char
=
random.choice([random_num, random_low_alpha, random_upper_alpha])
return
random_char
def
getValidCodeImg(
self
):
# 获取一个Image对象,参数分别是RGB模式。宽150,高30,随机颜色
image
=
Image.new(
'RGB'
,(
self
.width,
self
.height),
self
.getRandomColor())
# 获取一个画笔对象,将图片对象传过去
draw
=
ImageDraw.Draw(image)
# 获取一个font字体对象参数是ttf的字体文件的目录,以及字体的大小
font
=
ImageFont.truetype(
"kumo.ttf"
,size
=
self
.font_size)
temp
=
[]
for
i
in
range
(
self
.code_count):
# 循环5次,获取5个随机字符串
random_char
=
self
.getRandomStr()
# 在图片上一次写入获得的随机字符串,参数是:定位,字符串,颜色,字体
draw.text((
10
+
i
*
30
,
-
2
),random_char ,
self
.getRandomColor(), font
=
font)
# 保存随机字符,以供验证用户输入的验证码是否正确时使用
temp.append(random_char)
valid_str
=
"".join(temp)
# 噪点噪线
# 划线
for
i
in
range
(
self
.line_count):
x1
=
random.randint(
0
,
self
.width)
x2
=
random.randint(
0
,
self
.width)
y1
=
random.randint(
0
,
self
.height)
y2
=
random.randint(
0
,
self
.height)
draw.line((x1,y1,x2,y2),fill
=
self
.getRandomColor())
# 画点
for
i
in
range
(
self
.point_count):
draw.point([random.randint(
0
,
self
.width), random.randint(
0
,
self
.height)], fill
=
self
.getRandomColor())
x
=
random.randint(
0
,
self
.width)
y
=
random.randint(
0
,
self
.height)
draw.arc((x, y, x
+
4
, y
+
4
),
0
,
90
, fill
=
self
.getRandomColor())
# 在内存生成图片
from
io
import
BytesIO
f
=
BytesIO()
image.save(f,
self
.img_format)
data
=
f.getvalue()
f.close()
return
data,valid_str
if
__name__
=
=
'__main__'
:
img
=
ValidCodeImg()
data, valid_str
=
img.getValidCodeImg()
print
(valid_str)
f
=
open
(
'test.png'
,
'wb'
)
f.write(data)
|
效果:字体
七、应用到实际开发中
login.htmlurl
1
2
|
<
input
id
=
"valid-inp"
name
=
"validcode"
class
=
"form-control"
type
=
"password"
placeholder
=
"请输入验证码"
autocomplete
=
"off"
>
<span
id
=
"valid-img"
><img
id
=
"img"
src
=
"/get_valid_img"
title
=
"点击再换一张"
alt
=
"验证码图片"
><
/
span>
|
urls.pyspa
1
2
3
4
5
6
7
8
9
10
11
12
13
|
from
django.conf.urls
import
url
from
django.contrib
import
admin
from
blog
import
views
urlpatterns
=
[
url(r
'^admin/'
, admin.site.urls),
url(r
'^$'
,views.Main.as_view(),name
=
'main'
),
url(r
'^login$'
,views.Login.as_view(),name
=
'login'
),
# 登陆页面验证码图片请求
url(r
'^get_valid_img'
,views.GetValidImg.as_view(),name
=
'get_valid_img'
),
]
|
views.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
|
class
Main(View):
def
get(
self
,request):
return
render(request,
'main.html'
)
class
Login(View):
def
get(
self
,request):
return
render(request,
'login.html'
)
def
post(
self
,request):
username
=
request.POST.get(
'username'
)
password
=
request.POST.get(
'password'
)
valid_code
=
request.POST.get(
'valid_code'
)
# print(valid_code)
# print(request.session.get('valid_code'))
if
valid_code.upper() !
=
request.session.get(
'valid_code'
).upper():
return
JsonResponse({
'state'
:
False
,
'msg'
:
'验证码错误'
})
user
=
auth.authenticate(request,username
=
username,password
=
password)
if
user:
# 登陆成功,经过auth的login方法将用户写到session中
auth.login(request,user)
# 提交表单登陆成功后跳转到用户本身的博客首页
redirect_url
=
'/{}'
.
format
(user.username)
return
JsonResponse({
'state'
:
True
,
'msg'
:
'登陆成功!'
,
'url'
:redirect_url})
else
:
return
JsonResponse({
'state'
:
False
,
'msg'
:
'用户名或密码错误!'
})
class
GetValidImg(View):
def
get(
self
,request):
obj
=
ValidCodeImg()
img_data,valid_code
=
obj.getValidCodeImg()
request.session[
'valid_code'
]
=
valid_code
return
HttpResponse(img_data)
|