侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

在Python web中实现验证码图片代码分享

Python  /  管理员 发布于 7年前   178

系统版本: CentOS 7.4
Python版本: Python 3.6.1

在现在的WEB中,为了防止爬虫类程序提交表单,图片验证码是最常见也是最简单的应对方法之一。

1.验证码图片的生成

  在python中,图片验证码一般用PIL或者Pillow库实现,下面就是利用Pillow生成图片验证码的代码:

#!/usr/bin/env python3#- * -coding: utf - 8 - * -#@Author: Yang#@ Time: 2017 / 11 / 06 1: 04import randomfrom PILimport Image, ImageDraw, ImageFont, ImageFilter_letter_cases = "abcdefghjkmnpqrstuvwxy"#小写字母, 去除可能干扰的i, l, o, z_upper_cases = _letter_cases.upper()# 大写字母_numbers = ''.join(map(str, range(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 = (230, 230, 230),    fg_color = (18, 18, 18),    font_size = 20,    font_type = ‘/usr/share / fonts / dejavu / DejaVuSans - Bold.ttf',    length = 4,    draw_lines = True,    n_line = (1, 2),    draw_points = True,    point_chance = 1):  '''@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 range(width):  for h in range(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, strsif __name__ == '__main__':  img, str = create_validate_code()img.save('./test.gif', 'gif')

  最后的结果会返回一个元组,第一个返回值为一个Image类的实例,第二个返回值为验证码图片中的字符串,可以用于比对验证码是否正确。

生成的验证码图片效果:

 但是需要注意一点,以上代码需要依赖于系统字体,如果 font_type设置不正确,就会抛出 OSError 异常。

对于CenOS系统,字体文件一般在 /usr/share/fonts/dejavu/ 下, 如CentOS 7.4:

从中随意选取一个即可。windows 下同理,只需将 font_type 设置成正确的字体路径即可, 如

font_type=r"C:\Windows\Fonts\Arial.ttf"

2.如何在网页中显示验证码

  在上述代码中,验证码都是以文件的方式保存。如果要在web中使用验证码,不可能每次都先生成验证码图片,先保存到磁盘,再返回给前端 web。这样会增加磁盘的开销,另外频繁产生的验证码也会占用大量的磁盘空间。这时,可以使用 BytesIO 模块,使验证码图片的读写直接在内存中进行,并直接返回给前端。同时将正确验证码字符串存在session中,当用户提交表单时,就可以和session中的正确字符串作比较了。

  以Flask为例,以下为在Flask中使用验证码的完整 Demo:

#!/usr/bin/env python3# -*- coding: utf-8 -*- # @Author : Yang# @Time  : 2017/11/08 15:35 import randomfrom PIL import Image, ImageDraw, ImageFont, ImageFilterfrom io import BytesIOfrom flask import Flask, session, request_letter_cases = "abcdefghjkmnpqrstuvwxy" # 小写字母,去除可能干扰的i,l,o,z_upper_cases = _letter_cases.upper() # 大写字母_numbers = ''.join(map(str, range(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=(230, 230, 230), fg_color=(18, 18, 18), font_size=20, font_type='/usr/share/fonts/dejavu/DejaVuSans-Bold.ttf', length=4, draw_lines=True, n_line=(1, 2), draw_points=True, point_chance=1):  '''  @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 range(width):      for h in range(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, strsapp = Flask(__name__)app.config.update(  DEBUG=True,  SECRET_KEY='...')@app.route('/')def index():  return 'test'@app.route('/code')def get_code():  # 把strs发给前端,或者在后台使用session保存  code_img, strs = create_validate_code()  buf = BytesIO()  code_img.save(buf, 'jpeg')  buf_str = buf.getvalue()  response = app.make_response(buf_str)  response.headers['Content-Type'] = 'image/gif'  session['img'] = strs.upper()  return [email protected]("/login", methods=["POST", "GET"])def login():  if request.method == 'POST':    if session.get('img') == request.form.get('img').upper():      return 'OK'    return 'Error'  return """  <form action="" method="post">    <p>Name:<input type=text name=username>    <p>Password:<input type=text name=password>    <p>CAPTCHA:<input type=text name=img>    <img id="verficode" src="https:/article/./code" onclick="this.src='https:/article/./code?'+Math.random()">    # onclick事件用于每次点击时获取一个新的验证码    <p><input type=submit value=Login>  </form>  """if __name__ == "__main__":  app.run(host="0.0.0.0", port=18888, debug=True)

最终效果:

总结

以上就是本文关于在Python web中实现验证码图片代码分享的全部内容,希望对大家有所帮助。感兴趣的朋友可以继续参阅本站:python实现人脸识别代码、Python爬虫实例爬取网站搞笑段子、Python入门之三角函数全解【收藏】等,有什么问题可以随时留言,小编会及时回复大家的。感谢朋友们对本站的支持!


  • 上一条:
    python中Switch/Case实现的示例代码
    下一条:
    Python模糊查询本地文件夹去除文件后缀的实例(7行代码)
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • 在python语言中Flask框架的学习及简单功能示例(0个评论)
    • 在Python语言中实现GUI全屏倒计时代码示例(0个评论)
    • Python + zipfile库实现zip文件解压自动化脚本示例(0个评论)
    • python爬虫BeautifulSoup快速抓取网站图片(1个评论)
    • vscode 配置 python3开发环境的方法(0个评论)
    • 近期文章
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf文件功能(0个评论)
    • Laravel从Accel获得5700万美元A轮融资(0个评论)
    • 在go + gin中gorm实现指定搜索/区间搜索分页列表功能接口实例(0个评论)
    • 在go语言中实现IP/CIDR的ip和netmask互转及IP段形式互转及ip是否存在IP/CIDR(0个评论)
    • PHP 8.4 Alpha 1现已发布!(0个评论)
    • Laravel 11.15版本发布 - Eloquent Builder中添加的泛型(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2016-10
    • 2016-11
    • 2018-04
    • 2020-03
    • 2020-04
    • 2020-05
    • 2020-06
    • 2022-01
    • 2023-07
    • 2023-10
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客