Python编写生成验证码的脚本的教程


Posted in Python onMay 04, 2015

在web开发中经常用到验证码,为了防止机器人注册或者恶意登陆和查询等,作用不容小觑

但是验证码其实不是一个函数就能搞定的,它需要生成图片和水印,其实每种语言都有相关的函数生成图片和文字水印。包括我熟悉的php,呵呵,今天主要来分享如何用python生成验证码。

python生成验证码主要用到如下模块:Image, ImageDraw, ImageFont, ImageFilter和随机数生成模块Random。

代码如下:

#!/usr/bin/env python
#coding=utf-8
import random
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))
fontType="/usr/share/fonts/truetype/freefont/FreeSans.ttf"
 
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=fontType,
               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) # 创建画笔
 if draw_lines:
  create_lines(draw,n_line,width,height)
 if draw_points:
  create_points(draw,point_chance,width,height)
 strs = create_strs(draw,chars,length,font_type, font_size,width,height,fg_color)
 
 # 图形扭曲参数
 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
 
 
def create_lines(draw,n_line,width,height):
 '''绘制干扰线'''
 line_num = random.randint(n_line[0],n_line[1]) # 干扰线条数
 for i in range(line_num):
  # 起始点
  begin = (random.randint(0, width), random.randint(0, height))
  #结束点
  end = (random.randint(0, width), random.randint(0, height))
  draw.line([begin, end], fill=(0, 0, 0))
 
def create_points(draw,point_chance,width,height):
 '''绘制干扰点'''
 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(draw,chars,length,font_type, font_size,width,height,fg_color):
 '''绘制验证码字符'''
 '''生成给定长度的字符串,返回列表格式'''
 c_chars = random.sample(chars, length)
 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 __name__ == "__main__":
  code_img = create_validate_code()
  code_img[0].save("validate.gif", "GIF")
  print code_img[1]
Python 相关文章推荐
Python使用回溯法子集树模板解决爬楼梯问题示例
Sep 08 Python
python如何使用正则表达式的前向、后向搜索及前向搜索否定模式详解
Nov 08 Python
python将.ppm格式图片转换成.jpg格式文件的方法
Oct 27 Python
python 与服务器的共享文件夹交互方法
Dec 27 Python
对Xpath 获取子标签下所有文本的方法详解
Jan 02 Python
python实现在遍历列表时,直接对dict元素增加字段的方法
Jan 15 Python
Python3爬楼梯算法示例
Mar 04 Python
Django中celery执行任务结果的保存方法
Jul 12 Python
浅谈Python中(&,|)和(and,or)之间的区别
Aug 07 Python
python global关键字的用法详解
Sep 05 Python
django项目中新增app的2种实现方法
Apr 01 Python
Python flask路由间传递变量实例详解
Jun 03 Python
使用Python制作获取网站目录的图形化程序
May 04 #Python
使用Python脚本来获取Cisco设备信息的示例
May 04 #Python
用Python脚本来删除指定容量以上的文件的教程
May 04 #Python
编写Python脚本来获取Google搜索结果的示例
May 04 #Python
编写Python脚本来实现最简单的FTP下载的教程
May 04 #Python
Python下线程之间的共享和释放示例
May 04 #Python
简单介绍Python中利用生成器实现的并发编程
May 04 #Python
You might like
php完全过滤HTML,JS,CSS等标签
2009/01/16 PHP
php设计模式 Observer(观察者模式)
2011/06/26 PHP
php define的第二个参数使用方法
2013/11/04 PHP
PHP统计当前在线用户数实例讲解
2015/10/21 PHP
PHP实现合并两个排序链表的方法
2018/01/19 PHP
Javascript实现DIV滚动自动滚动到底部的代码
2012/03/01 Javascript
JavaScript学习笔记记录我的旅程
2012/05/23 Javascript
javascript控制swfObject应用介绍
2012/11/29 Javascript
Nodejs极简入门教程(三):进程
2014/10/27 NodeJs
js函数与php函数的区别实例浅析
2015/01/12 Javascript
浅析javascript中函数声明和函数表达式的区别
2015/02/15 Javascript
Web打印解决方案之证件套打的实现思路
2016/08/29 Javascript
jQuery实现的瀑布流加载效果示例
2016/09/13 Javascript
Js动态设置rem来实现移动端字体的自适应代码
2016/10/14 Javascript
JavaScript定义函数_动力节点Java学院整理
2017/06/27 Javascript
详解Angular2学习笔记之Html属性绑定
2018/01/03 Javascript
angular.js和vue.js中实现函数去抖示例(debounce)
2018/01/18 Javascript
微信小程序block的使用教程
2018/04/01 Javascript
移动端底部导航固定配合vue-router实现组件切换功能
2019/06/13 Javascript
详解小程序如何动态绑定点击的执行方法
2019/11/26 Javascript
[01:56]2014DOTA2西雅图邀请赛 MVP外卡赛老队长精辟点评
2014/07/09 DOTA
[03:14]辉夜杯主赛事 12月25日每日之星
2015/12/26 DOTA
Python爬虫辅助利器PyQuery模块的安装使用攻略
2016/04/24 Python
python字符串过滤性能比较5种方法
2017/06/22 Python
在Python中获取两数相除的商和余数方法
2018/11/10 Python
简单了解django缓存方式及配置
2019/07/19 Python
在django中使用post方法时,需要增加csrftoken的例子
2020/03/13 Python
在终端启动Python时报错的解决方案
2020/11/20 Python
瑞士设计师家具和家居饰品网上商店:Bruno Wickart
2019/03/18 全球购物
实习求职信
2013/12/01 职场文书
家长学校实施方案
2014/03/15 职场文书
2014年会计工作总结
2014/11/27 职场文书
新郎新娘答谢词
2015/01/04 职场文书
合理化建议书
2015/02/04 职场文书
离婚案件答辩状
2015/05/22 职场文书
获奖感言范文
2015/07/31 职场文书