1、基於PIL生成一個帶驗證碼的圖片和驗證碼,生成驗證碼圖片需要Monaco.ttf字體,可按自己要求更改check_code中的字體和字體文件位置,如下圖

1 #!/usr/bin/env python 2 # -*- coding:utf-8 -*- 3 4 import random 5 from PIL import Image, ImageDraw, ImageFont, ImageFilter 6 7 _letter_cases = "abcdefghjkmnpqrstuvwxy" # 小寫字母,去除可能干擾的i,l,o,z 8 _upper_cases = _letter_cases.upper() # 大寫字母 9 _numbers = ''.join(map(str, range(3, 10))) # 數字 10 init_chars = ''.join((_letter_cases, _upper_cases, _numbers)) 11 12 13 def create_validate_code(size=(120, 30), 14 chars=init_chars, 15 img_type="GIF", 16 mode="RGB", 17 bg_color=(255, 255, 255), 18 fg_color=(0, 0, 255), 19 font_size=18, 20 font_type="Monaco.ttf", 21 length=4, 22 draw_lines=True, 23 n_line=(1, 2), 24 draw_points=True, 25 point_chance=2): 26 """ 27 @todo: 生成驗證碼圖片 28 @param size: 圖片的大小,格式(寬,高),默認為(120, 30) 29 @param chars: 允許的字符集合,格式字符串 30 @param img_type: 圖片保存的格式,默認為GIF,可選的為GIF,JPEG,TIFF,PNG 31 @param mode: 圖片模式,默認為RGB 32 @param bg_color: 背景顏色,默認為白色 33 @param fg_color: 前景色,驗證碼字符顏色,默認為藍色#0000FF 34 @param font_size: 驗證碼字體大小 35 @param font_type: 驗證碼字體,默認為 ae_AlArabiya.ttf 36 @param length: 驗證碼字符個數 37 @param draw_lines: 是否划干擾線 38 @param n_lines: 干擾線的條數范圍,格式元組,默認為(1, 2),只有draw_lines為True時有效 39 @param draw_points: 是否畫干擾點 40 @param point_chance: 干擾點出現的概率,大小范圍[0, 100] 41 @return: [0]: PIL Image實例 42 @return: [1]: 驗證碼圖片中的字符串 43 """ 44 45 width, height = size # 寬高 46 # 創建圖形 47 img = Image.new(mode, size, bg_color) 48 draw = ImageDraw.Draw(img) # 創建畫筆 49 50 def get_chars(): 51 """生成給定長度的字符串,返回列表格式""" 52 return random.sample(chars, length) 53 54 def create_lines(): 55 """繪制干擾線""" 56 line_num = random.randint(*n_line) # 干擾線條數 57 58 for i in range(line_num): 59 # 起始點 60 begin = (random.randint(0, size[0]), random.randint(0, size[1])) 61 # 結束點 62 end = (random.randint(0, size[0]), random.randint(0, size[1])) 63 draw.line([begin, end], fill=(0, 0, 0)) 64 65 def create_points(): 66 """繪制干擾點""" 67 chance = min(100, max(0, int(point_chance))) # 大小限制在[0, 100] 68 69 for w in range(width): 70 for h in range(height): 71 tmp = random.randint(0, 100) 72 if tmp > 100 - chance: 73 draw.point((w, h), fill=(0, 0, 0)) 74 75 def create_strs(): 76 """繪制驗證碼字符""" 77 c_chars = get_chars() 78 strs = ' %s ' % ' '.join(c_chars) # 每個字符前后以空格隔開 79 80 font = ImageFont.truetype(font_type, font_size) 81 font_width, font_height = font.getsize(strs) 82 83 draw.text(((width - font_width) / 3, (height - font_height) / 3), 84 strs, font=font, fill=fg_color) 85 86 return ''.join(c_chars) 87 88 if draw_lines: 89 create_lines() 90 if draw_points: 91 create_points() 92 strs = create_strs() 93 94 # 圖形扭曲參數 95 params = [1 - float(random.randint(1, 2)) / 100, 96 0, 97 0, 98 0, 99 1 - float(random.randint(1, 10)) / 100, 100 float(random.randint(1, 2)) / 500, 101 0.001, 102 float(random.randint(1, 2)) / 500 103 ] 104 img = img.transform(size, Image.PERSPECTIVE, params) # 創建扭曲 105 106 img = img.filter(ImageFilter.EDGE_ENHANCE_MORE) # 濾鏡,邊界加強(閾值更大) 107 108 return img, strs
2、創建urls和views,請按自己需求創建

1 # 將check_code包放在合適的位置,導入即可,我是放在utils下面 2 from utils import check_code 3 4 def create_code_img(request): 5 f = BytesIO() #直接在內存開辟一點空間存放臨時生成的圖片 6 7 img, code = check_code.create_validate_code() #調用check_code生成照片和驗證碼 8 request.session['check_code'] = code #將驗證碼存在服務器的session中,用於校驗 9 img.save(f,'PNG') #生成的圖片放置於開辟的內存中 10 return HttpResponse(f.getvalue()) #將內存的數據讀取出來,並以HttpResponse返回
urls我的設置:url(r'^create_code_img/', views.create_code_img)
3、前端應用驗證碼和點擊自動刷新

1 <div class="row"> 2 <div class="col-xs-7"> 3 <input type="text" class="form-control" name="check_code" id="check_code" placeholder="請輸入驗證碼"> 4 </div> 5 <div class="col-xs-5"> 6 <img id="check_code_img" src="/create_code_img/" onclick="refresh_check_code(this)"> 7 {# src是url路徑,可得到驗證碼圖片,點擊時調用refresh_check_code#} 8 </div> 9 </div>

1 <script> 2 function refresh_check_code(ths) { 3 ths.src += '?'; 4 {# src后面加問好會自動刷新驗證碼img的src#} 5 } 6 </script>
4、login的Views進行數據驗證,然后做相應的處理

post_check_code = request.POST.get('check_code') session_check_code = request.session['check_code'] if post_check_code.lower() == session_check_code.lower() : pass