Python使用PIL库实现验证码图片的方法


Posted in Python onMarch 11, 2016

本文实例讲述了Python使用PIL库实现验证码图片的方法。分享给大家供大家参考,具体如下:

现在的网页中,为了防止机器人提交表单,图片验证码是很常见的应对手段之一。这里就不详细介绍了,相信大家都遇到过。

现在就给出用Python的PIL库实现验证码图片的代码。代码中有详细注释。

#!/usr/bin/env python
#coding=utf-8
import random
from PIL import Image, ImageDraw, ImageFont, ImageFilter
_letter_cases = "abcdefghjkmnpqrstuvwxy" # 小写字母,去除可能干扰的i,l,o,z
_upper_cases = _letter_cases.upper() # 大写字母
_numbers = ''.join(map(str, range(3, 10))) # 数字
init_chars = ''.join((_letter_cases, _upper_cases, _numbers))
def create_validate_code(size=(120, 30),
             chars=init_chars,
             img_type="GIF",
             mode="RGB",
             bg_color=(255, 255, 255),
             fg_color=(0, 0, 255),
             font_size=18,
             font_type="ae_AlArabiya.ttf",
             length=4,
             draw_lines=True,
             n_line=(1, 2),
             draw_points=True,
             point_chance = 2):
  '''
  @todo: 生成验证码图片
  @param size: 图片的大小,格式(宽,高),默认为(120, 30)
  @param chars: 允许的字符集合,格式字符串
  @param img_type: 图片保存的格式,默认为GIF,可选的为GIF,JPEG,TIFF,PNG
  @param mode: 图片模式,默认为RGB
  @param bg_color: 背景颜色,默认为白色
  @param fg_color: 前景色,验证码字符颜色,默认为蓝色#0000FF
  @param font_size: 验证码字体大小
  @param font_type: 验证码字体,默认为 ae_AlArabiya.ttf
  @param length: 验证码字符个数
  @param draw_lines: 是否划干扰线
  @param n_lines: 干扰线的条数范围,格式元组,默认为(1, 2),只有draw_lines为True时有效
  @param draw_points: 是否画干扰点
  @param point_chance: 干扰点出现的概率,大小范围[0, 100]
  @return: [0]: PIL Image实例
  @return: [1]: 验证码图片中的字符串
  '''
  width, height = size # 宽, 高
  img = Image.new(mode, size, bg_color) # 创建图形
  draw = ImageDraw.Draw(img) # 创建画笔
  def get_chars():
    '''生成给定长度的字符串,返回列表格式'''
    return random.sample(chars, length)
  def create_lines():
    '''绘制干扰线'''
    line_num = random.randint(*n_line) # 干扰线条数
    for i in range(line_num):
      # 起始点
      begin = (random.randint(0, size[0]), random.randint(0, size[1]))
      #结束点
      end = (random.randint(0, size[0]), random.randint(0, size[1]))
      draw.line([begin, end], fill=(0, 0, 0))
  def create_points():
    '''绘制干扰点'''
    chance = min(100, max(0, int(point_chance))) # 大小限制在[0, 100]
    for w in xrange(width):
      for h in xrange(height):
        tmp = random.randint(0, 100)
        if tmp > 100 - chance:
          draw.point((w, h), fill=(0, 0, 0))
  def create_strs():
    '''绘制验证码字符'''
    c_chars = get_chars()
    strs = ' %s ' % ' '.join(c_chars) # 每个字符前后以空格隔开
    font = ImageFont.truetype(font_type, font_size)
    font_width, font_height = font.getsize(strs)
    draw.text(((width - font_width) / 3, (height - font_height) / 3),
          strs, font=font, fill=fg_color)
    return ''.join(c_chars)
  if draw_lines:
    create_lines()
  if draw_points:
    create_points()
  strs = create_strs()
  # 图形扭曲参数
  params = [1 - float(random.randint(1, 2)) / 100,
       0,
       0,
       0,
       1 - float(random.randint(1, 10)) / 100,
       float(random.randint(1, 2)) / 500,
       0.001,
       float(random.randint(1, 2)) / 500
       ]
  img = img.transform(size, Image.PERSPECTIVE, params) # 创建扭曲
  img = img.filter(ImageFilter.EDGE_ENHANCE_MORE) # 滤镜,边界加强(阈值更大)
  return img, strs
if __name__ == "__main__":
  code_img = create_validate_code()
  code_img.save("validate.gif", "GIF")

最后结果返回一个元组,第一个返回值是Image类的实例,第二个参数是图片中的字符串(比较是否正确的作用)。

最后结果返回一个元组,第一个返回值是Image类的实例,第二个参数是图片中的字符串(比较是否正确的作用)。

需要提醒的是,如果在生成ImageFont.truetype实例的时候抛出IOError异常,有可能是运行代码的电脑没有包含指定的字体,需要下载安装。

生成的验证码图片效果:

Python使用PIL库实现验证码图片的方法

这时候,细心的同学可能要问,如果每次生成验证码,都要先保存生成的图片,再显示到页面。这么做让人太不能接受了。这个时候,我们需要使用python内置的StringIO模块,它有着类似file对象的行为,但是它操作的是内存文件。于是,我们可以这么写代码:

try:
  import cStringIO as StringIO
except ImportError:
  import StringIO
mstream = StringIO.StringIO()
img = create_validate_code()[0]
img.save(mstream, "GIF")

这样,我们需要输出的图片的时候只要使用“mstream.getvalue()”即可。比如在Django里,我们首先定义这样的url:

from django.conf.urls.defaults import *
urlpatterns = patterns('example.views',
  url(r'^validate/$', 'validate', name='validate'),
)

在views中,我们把正确的字符串保存在session中,这样当用户提交表单的时候,就可以和session中的正确字符串进行比较。

from django.shortcuts import HttpResponse
from validate import create_validate_code
def validate(request):
  mstream = StringIO.StringIO()
  validate_code = create_validate_code()
  img = validate_code[0]
  img.save(mstream, "GIF")
  request.session['validate'] = validate_code[1]
  return HttpResponse(mstream.getvalue(), "image/gif")

希望本文所述对大家Python程序设计有所帮助。

Python 相关文章推荐
python django 访问静态文件出现404或500错误
Jan 20 Python
Python实现字典的遍历与排序功能示例
Dec 23 Python
Python多线程扫描端口代码示例
Feb 09 Python
python实现对指定字符串补足固定长度倍数截断输出的方法
Nov 15 Python
python实现飞机大战游戏
Oct 26 Python
Python实现随机取一个矩阵数组的某几行
Nov 26 Python
python实现人机猜拳小游戏
Feb 03 Python
python和js交互调用的方法
Jun 23 Python
Python爬虫爬取博客实现可视化过程解析
Jun 29 Python
Python Matplotlib绘制动画的代码详解
May 30 Python
python中filter,map,reduce的作用
Jun 10 Python
如何利用python实现列表嵌套字典取值
Jun 10 Python
Python2.x利用commands模块执行Linux shell命令
Mar 11 #Python
Python实现列表转换成字典数据结构的方法
Mar 11 #Python
python中enumerate函数遍历元素用法分析
Mar 11 #Python
python实现class对象转换成json/字典的方法
Mar 11 #Python
Windows下Python的Django框架环境部署及应用编写入门
Mar 10 #Python
深入学习python的yield和generator
Mar 10 #Python
Python中random模块生成随机数详解
Mar 10 #Python
You might like
php 文件缓存函数
2011/10/08 PHP
采用thinkphp自带方法生成静态html文件详解
2014/06/13 PHP
php flush无效,IIS7下php实时输出的方法
2016/08/25 PHP
laravel migrate初学常见错误的解决方法
2017/10/11 PHP
详解PHP实现支付宝小程序用户授权的工具类
2018/12/25 PHP
laravel 自定义常量的两种方案
2019/10/14 PHP
JavaScript的Cookies
2008/01/16 Javascript
页面中iframe相互传值传参
2009/12/13 Javascript
利用毫秒减值计算时长的js代码
2013/09/22 Javascript
浅谈js中调用函数时加不加括号的问题
2016/07/28 Javascript
Javascript 之封装(Package)
2018/09/14 Javascript
微信小程序自定义模态弹窗组件详解
2019/12/24 Javascript
node事件循环和process模块实例分析
2020/02/14 Javascript
javascript canvas检测小球碰撞
2020/04/17 Javascript
基于vue和bootstrap实现简单留言板功能
2020/05/30 Javascript
结合axios对项目中的api请求进行封装操作
2020/09/21 Javascript
[01:10:02]IG vs Winstrike 2018国际邀请赛小组赛BO2 第一场 8.19
2018/08/21 DOTA
[45:44]完美世界DOTA2联赛PWL S2 FTD vs PXG 第一场 11.27
2020/12/01 DOTA
Python实现从订阅源下载图片的方法
2015/03/11 Python
在Mac OS上搭建Python的开发环境
2015/12/24 Python
用Python写一段用户登录的程序代码
2018/04/22 Python
详解Python数据分析--Pandas知识点
2019/03/23 Python
Python实现随机取一个矩阵数组的某几行
2019/11/26 Python
python写一个随机点名软件的实例
2019/11/28 Python
window环境pip切换国内源(pip安装异常缓慢的问题)
2019/12/31 Python
德国体育用品网上商店:SC24.com
2016/08/01 全球购物
买卖正宗运动鞋:GOAT
2019/12/06 全球购物
荷兰家电购物网站:Expert.nl
2020/01/18 全球购物
新西兰最大的连锁超市:Countdown
2020/06/04 全球购物
上海微创软件面试题
2012/06/14 面试题
致400米运动员广播稿
2014/02/07 职场文书
护士医德医风自我评价
2014/09/15 职场文书
代办出身证明书
2014/10/21 职场文书
2014年高中教师工作总结
2014/12/19 职场文书
教师岗位职责范本
2015/04/02 职场文书
浅谈移动端中的视口(viewport)的具体使用
2021/04/13 HTML / CSS