Python生成圖片驗(yàn)證碼

最近公司網(wǎng)站,需要在注冊(cè)模塊添加驗(yàn)證碼,防止其他人頻繁的惡意注冊(cè),我們后端使用的是python進(jìn)行開(kāi)發(fā),所以研究了下python圖片驗(yàn)證碼的方法。

最后確定使用python里面PIL庫(kù),通過(guò)Image, ImageDraw, ImageFont, ImageFilter的模塊生成圖片驗(yàn)證碼

設(shè)計(jì)思路(這里就不畫(huà)圖了):

(1)用戶填寫(xiě)用戶名(必須先填)
(2)客戶端點(diǎn)擊獲取驗(yàn)證碼,請(qǐng)求里帶用戶名參數(shù)
(3)調(diào)用生成驗(yàn)證碼接口
(4)按一定規(guī)則將用戶名生成鍵值,以生成的鍵值為鍵,驗(yàn)證碼為值,保存到緩存中
(5)將驗(yàn)證碼的字節(jié)流返回客戶端并顯示
(6)輸入驗(yàn)證碼,點(diǎn)擊注冊(cè)
(7)后臺(tái)根據(jù)用戶名和驗(yàn)證碼驗(yàn)證

生成驗(yàn)證碼函數(shù)

#!/usr/bin/env python
# -*- coding: UTF-8 -*-

import random
from PIL import Image, ImageDraw, ImageFont, ImageFilter

try:
    import cStringIO as StringIO
except ImportError:
    import StringIO

_letter_cases = "abcdefghjkmnpqrstuvwxy"                        # 小寫(xiě)字母
_upper_cases = "ABCDEFGHJKLMNPQRSTUVWXY"                        # 大寫(xiě)字母
_numbers = "1234567890"                                         # 數(shù)字
init_chars = ''.join((_letter_cases, _upper_cases, _numbers))   # 生成允許的字符集合
default_font = "./DejaVuSans.ttf"                               # 驗(yàn)證碼字體


# 生成驗(yàn)證碼接口
def generate_verify_image(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=default_font,
                          length=4,
                          draw_lines=True,
                          n_line=(1, 2),
                          draw_points=True,
                          point_chance=2,
                          save_img=False):

    """
    生成驗(yàn)證碼圖片
    :param size: 圖片的大小,格式(寬,高),默認(rèn)為(120, 30)
    :param chars: 允許的字符集合,格式字符串
    :param img_type: 圖片保存的格式,默認(rèn)為GIF,可選的為GIF,JPEG,TIFF,PNG
    :param mode: 圖片模式,默認(rèn)為RGB
    :param bg_color: 背景顏色,默認(rèn)為白色
    :param fg_color: 前景色,驗(yàn)證碼字符顏色,默認(rèn)為藍(lán)色#0000FF
    :param font_size: 驗(yàn)證碼字體大小
    :param font_type: 驗(yàn)證碼字體,默認(rèn)為 DejaVuSans.ttf
    :param length: 驗(yàn)證碼字符個(gè)數(shù)
    :param draw_lines: 是否劃干擾線
    :param n_line: 干擾線的條數(shù)范圍,格式元組,默認(rèn)為(1, 2),只有draw_lines為T(mén)rue時(shí)有效
    :param draw_points: 是否畫(huà)干擾點(diǎn)
    :param point_chance: 干擾點(diǎn)出現(xiàn)的概率,大小范圍[0, 100]
    :param save_img: 是否保存為圖片
    :return: [0]: 驗(yàn)證碼字節(jié)流, [1]: 驗(yàn)證碼圖片中的字符串
    """

    width, height = size  # 寬, 高
    img = Image.new(mode, size, bg_color)  # 創(chuàng)建圖形
    draw = ImageDraw.Draw(img)  # 創(chuàng)建畫(huà)筆

    def get_chars():
        """生成給定長(zhǎng)度的字符串,返回列表格式"""

        return random.sample(chars, length)

    def create_lines():
        """繪制干擾線"""

        line_num = random.randint(*n_line)  # 干擾線條數(shù)

        for i in range(line_num):
            # 起始點(diǎn)
            begin = (random.randint(0, size[0]), random.randint(0, size[1]))
            # 結(jié)束點(diǎn)
            end = (random.randint(0, size[0]), random.randint(0, size[1]))
            draw.line([begin, end], fill=(0, 0, 0))

    def create_points():
        """繪制干擾點(diǎn)"""

        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():
        """繪制驗(yàn)證碼字符"""

        c_chars = get_chars()
        strs = ' %s ' % ' '.join(c_chars)  # 每個(gè)字符前后以空格隔開(kāi)

        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()

    # 圖形扭曲參數(shù)
    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)  # 創(chuàng)建扭曲

    img = img.filter(ImageFilter.EDGE_ENHANCE_MORE)  # 濾鏡,邊界加強(qiáng)(閾值更大)

    mstream = StringIO.StringIO()
    img.save(mstream, img_type)

    if save_img:
        img.save("validate.gif", img_type)

    return mstream, strs


if __name__ == "__main__":
    mstream, strs = generate_verify_image(save_img=True)
    print strs

注意:返回的流要進(jìn)行轉(zhuǎn)換,在返回前端

self.write(simplejson.dumps({'code': 0, 'img': stream.getvalue().encode('base64')}))
#這里是將stream的值進(jìn)行了一次base64的編碼

前端js設(shè)置圖片src代碼

$("#verify_code_img").attr("src", "data:image/gif;base64," + data.img);

效果

是不是很簡(jiǎn)單,下次分享前端生成圖片驗(yàn)證碼的方法

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 178,932評(píng)論 25 709
  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 136,537評(píng)論 19 139
  • 22年12月更新:個(gè)人網(wǎng)站關(guān)停,如果仍舊對(duì)舊教程有興趣參考 Github 的markdown內(nèi)容[https://...
    tangyefei閱讀 35,393評(píng)論 22 257
  • 最近總是下雨,大雨的滂沱將天地間的一切都隱埋在它那巨大黑色斗篷下,恍惚中只剩下雨聲和雷聲了,其他的一切仿佛都不曾存...
    繩結(jié)季事閱讀 672評(píng)論 0 0
  • 周一,我們迎來(lái)了新歸隊(duì)的家人,看著家人們的激情與澎湃,不免回想起我們培訓(xùn)的經(jīng)過(guò)。是的,有些時(shí)候,過(guò)去了似乎就...
    書(shū)_贏閱讀 214評(píng)論 0 1

友情鏈接更多精彩內(nèi)容