生成随机验证码

浏览目录

Python生成随机验证码,需要使用PIL模块.python3则是pillow

安装:

pip install pillow

  

基本使用

1、创建图片

from PIL import Image
img = Image.new(mode='RGB', size=(120, 30), color=(255, 255, 255))
  
# 在图片查看器中打开
# img.show()
  
# 保存在本地
with open('code.png','wb') as f:
    img.save(f,format='png')  

2、创建画笔,用于在图片上画任意内容

img = Image.new(mode='RGB', size=(120, 30), color=(255, 255, 255))
draw = ImageDraw.Draw(img, mode='RGB') 

3、画点

img = Image.new(mode='RGB', size=(120, 30), color=(255, 255, 255))
draw = ImageDraw.Draw(img, mode='RGB')
# 第一个参数:表示坐标
# 第二个参数:表示颜色
draw.point([100, 100], fill="red")
draw.point([300, 300], fill=(255, 255, 255)) 

4、画线

img = Image.new(mode='RGB', size=(120, 30), color=(255, 255, 255))
draw = ImageDraw.Draw(img, mode='RGB')
# 第一个参数:表示起始坐标和结束坐标
# 第二个参数:表示颜色
draw.line((100,100,100,300), fill='red')
draw.line((100,100,300,100), fill=(255, 255, 255)) 

5、画圆

img = Image.new(mode='RGB', size=(120, 30), color=(255, 255, 255))
draw = ImageDraw.Draw(img, mode='RGB')
# 第一个参数:表示起始坐标和结束坐标(圆要画在其中间)
# 第二个参数:表示开始角度
# 第三个参数:表示结束角度
# 第四个参数:表示颜色
draw.arc((100,100,300,300),0,90,fill="red")  

6、写文本

img = Image.new(mode='RGB', size=(120, 30), color=(255, 255, 255))
draw = ImageDraw.Draw(img, mode='RGB')
# 第一个参数:表示起始坐标
# 第二个参数:表示写入内容
# 第三个参数:表示颜色
draw.text([0,0],'python',"red")

7、特殊字体文字

img = Image.new(mode='RGB', size=(120, 30), color=(255, 255, 255))
draw = ImageDraw.Draw(img, mode='RGB')
# 第一个参数:表示字体文件路径
# 第二个参数:表示字体大小
font = ImageFont.truetype("kumo.ttf", 28)
# 第一个参数:表示起始坐标
# 第二个参数:表示写入内容
# 第三个参数:表示颜色
# 第四个参数:表示颜色
draw.text([0, 0], 'python', "red", font=font)  

图片验证码

import random
  
def check_code(width=120, height=30, char_length=5, font_file='kumo.ttf', font_size=28):
    code = []
    img = Image.new(mode='RGB', size=(width, height), color=(255, 255, 255))
    draw = ImageDraw.Draw(img, mode='RGB')
  
    def rndChar():
        """
        生成随机字母  
        :return:
        """
        return chr(random.randint(65, 90))
  
    def rndColor():
        """
        生成随机颜色
        :return:
        """
        return (random.randint(0, 255), random.randint(10, 255), random.randint(64, 255))
  
    # 写文字
    font = ImageFont.truetype(font_file, font_size)
    for i in range(char_length):
        char = rndChar()
        code.append(char)
        h = random.randint(0, 4)
        draw.text([i * width / char_length, h], char, font=font, fill=rndColor())
  
    # 写干扰点
    for i in range(40):
        draw.point([random.randint(0, width), random.randint(0, height)], fill=rndColor())
  
    # 写干扰圆圈
    for i in range(40):
        draw.point([random.randint(0, width), random.randint(0, height)], fill=rndColor())
        x = random.randint(0, width)
        y = random.randint(0, height)
        draw.arc((x, y, x + 4, y + 4), 0, 90, fill=rndColor())
  
    # 画干扰线
    for i in range(5):
        x1 = random.randint(0, width)
        y1 = random.randint(0, height)
        x2 = random.randint(0, width)
        y2 = random.randint(0, height)
  
        draw.line((x1, y1, x2, y2), fill=rndColor())
  
    img = img.filter(ImageFilter.EDGE_ENHANCE_MORE)
    return img,''.join(code)
  
  
if __name__ == '__main__':
    # 1. 直接打开
    # img,code = check_code()
    # img.show()
  
    # 2. 写入文件
    # img,code = check_code()
    # with open('code.png','wb') as f:
    #     img.save(f,format='png')
  
    # 3. 写入内存(Python3)
    # from io import BytesIO
    # stream = BytesIO()
    # img.save(stream, 'png')
    # stream.getvalue()
  
    # 4. 写入内存(Python2)
    # import StringIO
    # stream = StringIO.StringIO()
    # img.save(stream, 'png')
    # stream.getvalue()
  
    pass
def validcode(request):
    import random
    from PIL import Image, ImageDraw, ImageFont
    from io import BytesIO

    img = Image.new(mode='RGB', size=(120, 40),
                    color=(random.randint(0, 255), random.randint(0, 255), random.randint(0, 255)))
    draw = ImageDraw.Draw(img, 'RGB')
    font = ImageFont.truetype('app01/static/font/kumo.ttf', 25)
    valid_list = []
    for i in range(5):
        random_num = str(random.randint(0, 9))
        random_upper_alp = chr(random.randint(65, 90))
        random_lower_alp = chr(random.randint(97, 122))
        valid_ele = random.choice([random_num, random_upper_alp, random_lower_alp])
        valid_list.append(valid_ele)
        draw.text([5 + i * 24, 10], valid_ele, (random.randint(0, 255), random.randint(0, 255), random.randint(0, 255)),
                  font=font)

        # 写干扰点
    for i in range(40):
        draw.point([random.randint(0, 120), random.randint(0, 40)],
                   fill=(random.randint(0, 255), random.randint(10, 255), random.randint(64, 255)))

        # 写干扰圆圈
    for i in range(40):
        draw.point([random.randint(0, 120), random.randint(0, 40)],
                   fill=(random.randint(0, 255), random.randint(10, 255), random.randint(64, 255)))
        x = random.randint(0, 120)
        y = random.randint(0, 40)
        draw.arc((x, y, x + 4, y + 4), 0, 90,
                 fill=(random.randint(0, 255), random.randint(10, 255), random.randint(64, 255)))

        # 画干扰线
    for i in range(5):
        x1 = random.randint(0, 120)
        y1 = random.randint(0, 40)
        x2 = random.randint(0, 120)
        y2 = random.randint(0, 40)

        draw.line((x1, y1, x2, y2), fill=(random.randint(0, 255), random.randint(10, 255), random.randint(64, 255)))

    f = BytesIO()
    img.save(f, 'png')
    data = f.getvalue()
    valid_str = ''.join(valid_list)     #将验证码拼接成字符串
    print(valid_str)
    request.session["keepValidCode"] = valid_str   #将验证码保存在session中,方便在登陆时验证


    return HttpResponse(data)
def validcode(request)
from app01 import models
from django.contrib.auth import authenticate, login, logout
def log_in(request):
    '''登陆'''
    if request.method == 'GET':
        form = LoginForm()
        return render(request, 'login.html', {'form': form})
    else:
        data = request.POST
        form = LoginForm(data=data)
        validcode = request.POST.get('validcode')

        if not form.is_valid():
            return HttpResponse(json.dumps(dict(form.errors)))
        if validcode.upper() != request.session["keepValidCode"].upper():
            return HttpResponse(json.dumps('codeError'))

        user = authenticate(**form.cleaned_data)
        # user=models.UserInfo.objects.filter(**form.cleaned_data)
        if not user:

            return HttpResponse(json.dumps({'flag': False, 'msg': "用户名或者密码错误"}))

        else:
            login(request, user)  #设置session 
        return HttpResponse(json.dumps({'flag': True}))
def log_in(request)  
# 登录获取验证码
def get_valid_img(request):
    # 方式一
    with open("mj.jpg", "rb") as f:
        data = f.read()

    # 方式二
    # import PIL
    # from PIL import Image
    #
    # image = Image.new("RGB", (250, 40), "green")  #图片模式,大小,颜色
    #
    # with open("valid_code.png", "wb") as f:
    #     image.save(f, "png")
    # with open("valid_code.png", "rb") as f:
    #     data = f.read()



    # 方式三

    # import PIL
    # from PIL import Image
    # import random
    #
    #
    # def get_random_color():  # 随机颜色
    #     return (random.randint(0, 255), random.randint(0, 255), random.randint(0, 255))
    #
    #
    # image = Image.new("RGB", (250, 40), get_random_color())
    #
    # with open("valid_code.png", "wb") as f:
    #     image.save(f, "png")
    # with open("valid_code.png", "rb") as f:
    #     data = f.read()



    # 方式四
    # from PIL import Image
    # import random
    #
    # def get_random_color():  # 随机颜色
    #     return (random.randint(0, 255), random.randint(0, 255), random.randint(0, 255))
    #
    # image = Image.new("RGB", (250, 40), get_random_color())
    #生成的图片保存在内存
    # from io import BytesIO
    # f = BytesIO()
    # image.save(f, "png") #生成的图片保存在内存,png的格式
    # data = f.getvalue() #取出来
    # f.close()

    #方式五
    from PIL import Image
    from PIL import ImageDraw,ImageFont
    import random

    def get_random_color():  # 随机颜色
        return (random.randint(0, 255), random.randint(0, 255), random.randint(0, 255)) #红,绿,蓝

    image = Image.new("RGB", (250, 40), get_random_color()) #图片模式,大小,颜色

    draw=ImageDraw.Draw(image)  #在图片上写内容
    font=ImageFont.truetype("blog/static/font/kumo.ttf",size=32) #设置字体样式和大小
    temp=[]  #空列表保存生成的随机字符
    # 生成五个随机字符
    for i in range(5):
        random_num=str(random.randint(0,9)) #生成1——9de 随机数  int转成 str
        random_low_alpha=chr(random.randint(97,122)) #生成a——z的随机小写字母
        random_upper_alpha=chr(random.randint(65,90))#生成A——Z的随机大写字母
        random_char=random.choice([random_num,random_low_alpha,random_upper_alpha]) #数字,小写字母,大写字母,随机挑选
        #在生成的验证码图片添加内容样式
        #坐标,文本内容,图片颜色,字体
        draw.text((24+i*36,0),random_char,get_random_color(),font=font)
         # 生成的验证码内容添加到列表中保存,方便登录时校验
        temp.append(random_char)

        # 噪点噪线
        width=250
        height=40
        for i in range(2):
            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=get_random_color())

        for i in range(2):
            draw.point([random.randint(0, width), random.randint(0, height)], fill=get_random_color())
            x = random.randint(0, width)
            y = random.randint(0, height)
            draw.arc((x, y, x + 4, y + 4), 0, 90, fill=get_random_color())

    #在内存中生成图片
    from io import BytesIO
    f = BytesIO()
    image.save(f, "png") #生成的图片保存在内存,png的格式
    data = f.getvalue() #取出来
    f.close()

    valid_str="".join(temp) #列表转换成字符串,与文本框输入的内容作比较

    # session将验证码保存在数据库
    request.session["valid_str"]=valid_str

    return HttpResponse(data)   #返回验证码内容
随机验证码

img中str实现局部刷新验证码

html

<p><label class="lbright">验证码:</label>
<span>
<input type="text" name="validcode" style="70px; vertical-align:middle;" id="validcode"/>
<img id="codePic" src="http://127.0.0.1:8888/TP/codePic" width="60" height="21" style="vertical-align:middle;cursor:pointer;"/> 
</span> 
<a class="blurry" id="newPic" onclick="getPic();">看不清楚,换一张</a>
</p> 

js

<script type="text/javascript">
function getPic(){ 
$("#codePic").attr("src","http://127.0.0.1:8888/TP/codePic?flag="+Math.random()); 
};
</script>  

// 验证码刷新
    $(".validCode_img").click(function () {
        $(this)[0].src+="?"
    })

  

这部分最重要的就是 $("#codePic").attr("src","http://127.0.0.1:8888/TP/codePic?flag="+Math.random()); 这部分的代码。如果不加flag="+Math.random()是实现不了局部刷新的功能的。因为src中如果每次访问的地址一样的话就会发生不更新的情况。具体为什么会发生这种情况大家可以自己去研究。而codePic其实是一个action。这个action的功能是利用java画笔画出验证码并打包成图片返回给img中的src。

原文地址:https://www.cnblogs.com/gaoya666/p/8818770.html