17 python 之利用 pil 庫實現頁麵的圖片驗證碼及縮略圖
Python實戰-從菜鳥到大牛的進階之路 作者:極客學院 投票推薦 加入書簽 留言反饋
<h2>前言:</h2>
在域用戶平台裏麵集成了一個信使的功能,但是怕別人無腦的亂發,又怕別人亂調用接口,加了一個圖片驗證碼的功能~
實現起來比較簡單,用 python 的 pil 庫就可以實現了。
圖片 17.1 pic
yum install -y libjpeg-develyum install freetype-* </pre>
安裝 pil 庫 ~
圖片 17.2 pic
能不能用,要看下麵是否支持 jpeg,不然圖片沒法寫入的
圖片 17.3 pic
好嘞,咱們可以跑一個 demo 了~
import image, imagefont, imagedrawtext = "ewwiieat"im = image.new("rgb",(130,35), (255, 255, 255))dr = imagedraw.draw(im)font = imagefont.truetype("kk.ttf", 24) #simsunb.ttf 這個從windows fonts copy一個過來dr.text((10, 5), text, font=font, fill="#000000")im.showim.save("t.png") </pre>
圖片 17.4 pic
因為用的是終端,沒法看,傳到 windows 上,看下結果~
圖片 17.5 pic
pil 可以做很多的事情,比如縮略圖的實現~
from pil import imageimg = image.open(\''god.jpg'')img = img.resize((250, 156), image.antialias)img.save(\''sharejs_small.jpg'') </pre>
圖片 17.6 pic
對於驗證碼來說,上麵的有點太正派了,很容易被破解了,所以咱們把驗證碼的圖片給偽裝下。
#!/usr/bin/env python #coding=utf-8import randomfrom pil 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))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="kk.ttf", 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_rabiya.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 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:\''\''\''繪製驗證碼字符\''\''\''c_chars = get_charsstrs = \'' %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__": code_img = create_validate_code code_img[0].save("xiaorui.gif", "gif") </pre>
看下驗證碼效果 ~
圖片 17.7 pic
要是在 web 框架裏麵的話,可以直接 return 一個驗證碼圖片
比如:
tornado 的實現方式
output = stringioim.save(output, \''jpeg\'', quality = 95)img_data = output.getvalueoutput.close #在瀏覽器現實圖片set_header(\''content-type\'',\''image/jpeg\'')write(img_data) </pre>
我這裏用的是 sk,代碼實現是
@app.route(\''/pic\'')def showpic: import image, imagefont, imagedraw import stringio output = stringio.stringio text = "ewwiieat" im = image.new("rgb",(130,35), (255, 255, 255)) dr = imagedraw.draw(im) font = imagefont.truetype("kk.ttf", 24) #simsunb.ttf 這個從windows fonts copy一個過來 dr.text((10, 5), text, font=font, fill="#000000") # im.show im.save(output,"gif") img_data = output.getvalue output.close response = make_response(img_data) response.headers[\''content-type\''] = \''image/gif\'' return response </pre>
圖片 17.8 pic
本文出自 “峰雲,就她了。” 博客,謝絕轉載!
在域用戶平台裏麵集成了一個信使的功能,但是怕別人無腦的亂發,又怕別人亂調用接口,加了一個圖片驗證碼的功能~
實現起來比較簡單,用 python 的 pil 庫就可以實現了。
圖片 17.1 pic
yum install -y libjpeg-develyum install freetype-* </pre>
安裝 pil 庫 ~
圖片 17.2 pic
能不能用,要看下麵是否支持 jpeg,不然圖片沒法寫入的
圖片 17.3 pic
好嘞,咱們可以跑一個 demo 了~
import image, imagefont, imagedrawtext = "ewwiieat"im = image.new("rgb",(130,35), (255, 255, 255))dr = imagedraw.draw(im)font = imagefont.truetype("kk.ttf", 24) #simsunb.ttf 這個從windows fonts copy一個過來dr.text((10, 5), text, font=font, fill="#000000")im.showim.save("t.png") </pre>
圖片 17.4 pic
因為用的是終端,沒法看,傳到 windows 上,看下結果~
圖片 17.5 pic
pil 可以做很多的事情,比如縮略圖的實現~
from pil import imageimg = image.open(\''god.jpg'')img = img.resize((250, 156), image.antialias)img.save(\''sharejs_small.jpg'') </pre>
圖片 17.6 pic
對於驗證碼來說,上麵的有點太正派了,很容易被破解了,所以咱們把驗證碼的圖片給偽裝下。
#!/usr/bin/env python #coding=utf-8import randomfrom pil 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))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="kk.ttf", 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_rabiya.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 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:\''\''\''繪製驗證碼字符\''\''\''c_chars = get_charsstrs = \'' %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__": code_img = create_validate_code code_img[0].save("xiaorui.gif", "gif") </pre>
看下驗證碼效果 ~
圖片 17.7 pic
要是在 web 框架裏麵的話,可以直接 return 一個驗證碼圖片
比如:
tornado 的實現方式
output = stringioim.save(output, \''jpeg\'', quality = 95)img_data = output.getvalueoutput.close #在瀏覽器現實圖片set_header(\''content-type\'',\''image/jpeg\'')write(img_data) </pre>
我這裏用的是 sk,代碼實現是
@app.route(\''/pic\'')def showpic: import image, imagefont, imagedraw import stringio output = stringio.stringio text = "ewwiieat" im = image.new("rgb",(130,35), (255, 255, 255)) dr = imagedraw.draw(im) font = imagefont.truetype("kk.ttf", 24) #simsunb.ttf 這個從windows fonts copy一個過來 dr.text((10, 5), text, font=font, fill="#000000") # im.show im.save(output,"gif") img_data = output.getvalue output.close response = make_response(img_data) response.headers[\''content-type\''] = \''image/gif\'' return response </pre>
圖片 17.8 pic
本文出自 “峰雲,就她了。” 博客,謝絕轉載!