
作者:Kevin
來(lái)源:麥?zhǔn)寰幊?
最近感覺(jué)被大數(shù)據(jù)定義成機(jī)器人了,隨便看個(gè)網(wǎng)頁(yè)都跳驗(yàn)證碼。
怎么用python繞驗(yàn)證碼是個(gè)令人頭禿的事情,
我投降!那么今天手把手教大家如何寫(xiě)驗(yàn)證碼,去為難別人,讓他們頭禿。
說(shuō)錯(cuò)了,其實(shí)就是教大家如何通過(guò)python代碼去生成驗(yàn)證碼~~
pip install pillow
# 導(dǎo)入相關(guān)的繪畫(huà)模塊 from PIL import Image, ImageDraw, ImageFont # 設(shè)置背景色 bg_color = (100, 100, 255) #設(shè)置畫(huà)布長(zhǎng)寬(像素) width = 400 height = 100 # 通過(guò)設(shè)置生成新的畫(huà)布 im = Image.new('RGB',(width,height),bg_color) # 展示畫(huà)布 im.show()
在這bg_color背景色的設(shè)置是用RGB顏色標(biāo)準(zhǔn)去設(shè)置的,如果你不喜歡這個(gè)背景色可以自己調(diào)一下。
“
常見(jiàn)的RGB顏色
運(yùn)行代碼后:
先上代碼
from PIL import Image, ImageDraw, ImageFont # 省略第一步的代碼 # 創(chuàng)建畫(huà)筆對(duì)象 draw = ImageDraw.Draw(im) # 驗(yàn)證碼文本 string = 'MSBC' # 構(gòu)造字體對(duì)象 font = ImageFont.truetype('./ziti.ttf', 90) # font = ImageFont.load_default().font # 構(gòu)造字體顏色 fontcolor = (255, 100, 100) # 繪制4個(gè)字 draw.text((10, 2), string[0], font=font, fill=fontcolor)
draw.text((110, 2), string[1], font=font, fill=fontcolor)
draw.text((210, 2), string[2], font=font, fill=fontcolor)
draw.text((310, 2), string[3], font=font, fill=fontcolor) #釋放畫(huà)筆 del draw #展示圖片 im.show()
代碼分析:
draw = ImageDraw.Draw(im)
在im畫(huà)布上實(shí)例化一只筆。
font = ImageFont.truetype('./ziti.ttf', 90) # font = ImageFont.load_default().font
第一個(gè)參數(shù)是設(shè)置字體,我這有下載一個(gè)ttf的字體文件所以可以用它,如果沒(méi)有指定的字體文件可以使用默認(rèn)的# font = ImageFont.load_default().font;
第二個(gè)參數(shù)是繪制字體的大小,因?yàn)槲覀儺?huà)布是400x100的 所以我們?yōu)榱嗣烙^就把字體設(shè)成90x90的尺寸。
# 構(gòu)造字體顏色 fontcolor = (255, 100, 100)
字體文本的顏色,參照第一步畫(huà)布的RGB設(shè)置。
# 繪制4個(gè)字 draw.text((10, 2), string[0], font=font, fill=fontcolor)
draw.text((110, 2), string[1], font=font, fill=fontcolor)
draw.text((210, 2), string[2], font=font, fill=fontcolor)
draw.text((310, 2), string[3], font=font, fill=fontcolor)
這里draw.text函數(shù),顧名思義就是開(kāi)始拿畫(huà)起畫(huà)筆開(kāi)始寫(xiě)字,
第一個(gè)參數(shù) 寫(xiě)字的坐標(biāo);
第二個(gè)參數(shù) 要寫(xiě)的字;
第三個(gè)參數(shù) 字的顏色(上面構(gòu)造過(guò)了,你也可以設(shè)成一字一色)。
代碼跑一下看成果:
效果還行,就是總覺(jué)得少了點(diǎn)什么?
既然是驗(yàn)證碼,肯定要稍微難識(shí)別,上面那個(gè)那么傻白甜的驗(yàn)證碼是怎么回事??
這一步我們需要導(dǎo)入random模塊,因?yàn)楦蓴_是不規(guī)則隨機(jī)生成的。
import random from PIL import Image, ImageDraw, ImageFont # 省略第一步代碼 # 省略第二步代碼 #使用point函數(shù)繪制噪點(diǎn) for i in range(0, 100): xy = (random.randrange(0, width), random.randrange(0, height)) fill = (random.randrange(0, 255), 255, random.randrange(0, 255)) draw.point(xy, fill=fill) #釋放畫(huà)筆 del draw im.show()
代碼分析:
import random
別忘了導(dǎo)入random模塊
for i in range(0, 100): xy = (random.randrange(0, width), random.randrange(0, height)) fill = (255, 255, 255) draw.point(xy, fill=fill)
一個(gè)循環(huán)100次的for循環(huán),
xy變量是畫(huà)干擾點(diǎn)的坐標(biāo)值
fill變量是噪點(diǎn)的顏色,還是RGB標(biāo)準(zhǔn)的
draw.point 畫(huà)點(diǎn)的動(dòng)作
“
這個(gè)for循環(huán)的次數(shù)越多,畫(huà)布上噪點(diǎn)也會(huì)相應(yīng)增多。
跑一下代碼看看噪點(diǎn)的效果如何:
感覺(jué)還是有點(diǎn)傻白甜,我們來(lái)循環(huán)1000次的試試:
10000次!
夠了。
這一步我將各個(gè)參數(shù)結(jié)合random模塊,使我們的驗(yàn)證碼更難辨別!
import random from PIL import Image, ImageDraw, ImageFont
bg_color = (random.randrange(20, 120), random.randrange(20, 120), random.randrange(150, 255))
width = 400 height = 100 im = Image.new('RGB',(width,height),bg_color)
# 創(chuàng)建畫(huà)筆對(duì)象
draw = ImageDraw.Draw(im)
# 構(gòu)造字體對(duì)象
font = ImageFont.truetype('./ziti.ttf', 100)
# font = ImageFont.load_default().font
# 構(gòu)造字體顏色
fontcolor = (random.randrange(0, 255), random.randrange(0, 255), random.randrange(0, 255))
# 繪制4個(gè)字 string = 'MSBC' draw.text((random.randint(10, 30), (random.randint(0, 10))), string[0], font=font, fill=fontcolor)
draw.text((random.randint(90, 130), (random.randint(0, 10))), string[1], font=font, fill=fontcolor)
draw.text((random.randint(180, 230), (random.randint(0, 10))), string[2], font=font, fill=fontcolor)
draw.text((random.randint(270, 330), (random.randint(0, 10))), string[3], font=font, fill=fontcolor)
#調(diào)用畫(huà)筆的point()函數(shù)繪制噪點(diǎn) for i in range(0, 10000):
xy = (random.randrange(0, width), random.randrange(0, height))
fill = (random.randrange(0, 255), random.randrange(0, 255), random.randrange(0, 255))
draw.point(xy, fill=fill)
#釋放畫(huà)筆
del draw
im.show()
我把字體顏色,噪點(diǎn)顏色,文本位置都結(jié)合了random模塊,效果圖如下:
這一步,我們需要把上面的代碼封裝到函數(shù)中,大致把上面代碼重構(gòu)成:
# 使用for循環(huán)生成文本字符 # 生成驗(yàn)證碼圖片的函數(shù),參數(shù)就是上面生成的文本 # 調(diào)用生成驗(yàn)證碼圖片函數(shù)
重構(gòu)后:
import random from PIL import Image, ImageDraw, ImageFont string = '' #隨機(jī)選取4個(gè)值作為驗(yàn)證碼
rand_str = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789' for i in range(0, 4): string += rand_str[random.randrange(0, len(rand_str))]
def gen_verify_img(text):
bg_color = (random.randrange(20, 120), random.randrange(20, 120), random.randrange(150, 255))
width = 400 height = 100 im = Image.new('RGB',(width,height),bg_color)
# 創(chuàng)建畫(huà)筆對(duì)象
draw = ImageDraw.Draw(im)
# 構(gòu)造字體對(duì)象
font = ImageFont.truetype('./ziti.ttf', 100)
# font = ImageFont.load_default().font
# 構(gòu)造字體顏色
fontcolor = (random.randrange(0, 255), random.randrange(0, 255), random.randrange(0, 255))
# 繪制4個(gè)字
draw.text((random.randint(10, 30), (random.randint(0, 10))), string[0], font=font, fill=fontcolor)
draw.text((random.randint(90, 130), (random.randint(0, 10))), string[1], font=font, fill=fontcolor)
draw.text((random.randint(180, 230), (random.randint(0, 10))), string[2], font=font, fill=fontcolor)
draw.text((random.randint(270, 330), (random.randint(0, 10))), string[3], font=font, fill=fontcolor)
#調(diào)用畫(huà)筆的point()函數(shù)繪制噪點(diǎn) for i in range(0, 10000):
xy = (random.randrange(0, width), random.randrange(0, height))
fill = (random.randrange(0, 255), random.randrange(0, 255), random.randrange(0, 255))
draw.point(xy, fill=fill)
#釋放畫(huà)筆
del draw
im.show()
# 調(diào)用函數(shù)
gen_verify_img(string)
把原先代碼中的string變量提到了函數(shù)外,把它變成函數(shù)需要傳入的參數(shù),
再用for循環(huán),隨機(jī)選出4個(gè)字符。
string = '' #隨機(jī)選取4個(gè)值作為驗(yàn)證碼 rand_str = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789' for i in range(0, 4): string += rand_str[random.randrange(0, len(rand_str))]
代碼再跑一下:
上面的驗(yàn)證碼是 DZNO還是DZN0?
在web開(kāi)發(fā)的登錄操作,和訓(xùn)練驗(yàn)證碼識(shí)別的神經(jīng)運(yùn)算中,都需要大量的驗(yàn)證碼圖片。
所以需要把大量的驗(yàn)證碼圖片文件,我們將批量驗(yàn)證碼保存到本地。
完整代碼:
import random from PIL import Image, ImageDraw, ImageFont
def gen_verify_img(text):
bg_color = (random.randrange(20, 120), random.randrange(20, 120), random.randrange(150, 255))
width = 400 height = 100 im = Image.new('RGB',(width,height),bg_color)
# 創(chuàng)建畫(huà)筆對(duì)象
draw = ImageDraw.Draw(im)
# 構(gòu)造字體對(duì)象
font = ImageFont.truetype('./ziti.ttf', 100)
# font = ImageFont.load_default().font
# 構(gòu)造字體顏色
fontcolor = (random.randrange(0, 255), random.randrange(0, 255), random.randrange(0, 255))
# 繪制4個(gè)字
draw.text((random.randint(10, 30), (random.randint(0, 10))), string[0], font=font, fill=fontcolor)
draw.text((random.randint(90, 130), (random.randint(0, 10))), string[1], font=font, fill=fontcolor)
draw.text((random.randint(180, 230), (random.randint(0, 10))), string[2], font=font, fill=fontcolor)
draw.text((random.randint(270, 330), (random.randint(0, 10))), string[3], font=font, fill=fontcolor)
#調(diào)用畫(huà)筆的point()函數(shù)繪制噪點(diǎn) for i in range(0, 10000):
xy = (random.randrange(0, width), random.randrange(0, height))
fill = (random.randrange(0, 255), random.randrange(0, 255), random.randrange(0, 255))
draw.point(xy, fill=fill)
#釋放畫(huà)筆
del draw
# im.show()
im.save(f'./{text}.png','png') for i in range(100): string = '' #隨機(jī)選取4個(gè)值作為驗(yàn)證碼
rand_str = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789' for i in range(0, 4): string += rand_str[random.randrange(0, len(rand_str))]
gen_verify_img(string) print(f'{string} 驗(yàn)證碼生成成功!!')
最后再跑一下:
部分驗(yàn)證碼展示:
“
作者注:
如果再將本文中的代碼進(jìn)行變形或改寫(xiě),可能會(huì)得到更五花八門(mén)的驗(yàn)證碼,怎么發(fā)揮就看屏幕錢(qián)你的了。
數(shù)據(jù)分析咨詢請(qǐng)掃描二維碼
若不方便掃碼,搜微信號(hào):CDAshujufenxi
SQL Server 中 CONVERT 函數(shù)的日期轉(zhuǎn)換:從基礎(chǔ)用法到實(shí)戰(zhàn)優(yōu)化 在 SQL Server 的數(shù)據(jù)處理中,日期格式轉(zhuǎn)換是高頻需求 —— 無(wú)論 ...
2025-09-18MySQL 大表拆分與關(guān)聯(lián)查詢效率:打破 “拆分必慢” 的認(rèn)知誤區(qū) 在 MySQL 數(shù)據(jù)庫(kù)管理中,“大表” 始終是性能優(yōu)化繞不開(kāi)的話題。 ...
2025-09-18CDA 數(shù)據(jù)分析師:表結(jié)構(gòu)數(shù)據(jù) “獲取 - 加工 - 使用” 全流程的賦能者 表結(jié)構(gòu)數(shù)據(jù)(如數(shù)據(jù)庫(kù)表、Excel 表、CSV 文件)是企業(yè)數(shù)字 ...
2025-09-18DSGE 模型中的 Et:理性預(yù)期算子的內(nèi)涵、作用與應(yīng)用解析 動(dòng)態(tài)隨機(jī)一般均衡(Dynamic Stochastic General Equilibrium, DSGE)模 ...
2025-09-17Python 提取 TIF 中地名的完整指南 一、先明確:TIF 中的地名有哪兩種存在形式? 在開(kāi)始提取前,需先判斷 TIF 文件的類型 —— ...
2025-09-17CDA 數(shù)據(jù)分析師:解鎖表結(jié)構(gòu)數(shù)據(jù)特征價(jià)值的專業(yè)核心 表結(jié)構(gòu)數(shù)據(jù)(以 “行 - 列” 規(guī)范存儲(chǔ)的結(jié)構(gòu)化數(shù)據(jù),如數(shù)據(jù)庫(kù)表、Excel 表、 ...
2025-09-17Excel 導(dǎo)入數(shù)據(jù)含缺失值?詳解 dropna 函數(shù)的功能與實(shí)戰(zhàn)應(yīng)用 在用 Python(如 pandas 庫(kù))處理 Excel 數(shù)據(jù)時(shí),“缺失值” 是高頻 ...
2025-09-16深入解析卡方檢驗(yàn)與 t 檢驗(yàn):差異、適用場(chǎng)景與實(shí)踐應(yīng)用 在數(shù)據(jù)分析與統(tǒng)計(jì)學(xué)領(lǐng)域,假設(shè)檢驗(yàn)是驗(yàn)證研究假設(shè)、判斷數(shù)據(jù)差異是否 “ ...
2025-09-16CDA 數(shù)據(jù)分析師:掌控表格結(jié)構(gòu)數(shù)據(jù)全功能周期的專業(yè)操盤(pán)手 表格結(jié)構(gòu)數(shù)據(jù)(以 “行 - 列” 存儲(chǔ)的結(jié)構(gòu)化數(shù)據(jù),如 Excel 表、數(shù)據(jù) ...
2025-09-16MySQL 執(zhí)行計(jì)劃中 rows 數(shù)量的準(zhǔn)確性解析:原理、影響因素與優(yōu)化 在 MySQL SQL 調(diào)優(yōu)中,EXPLAIN執(zhí)行計(jì)劃是核心工具,而其中的row ...
2025-09-15解析 Python 中 Response 對(duì)象的 text 與 content:區(qū)別、場(chǎng)景與實(shí)踐指南 在 Python 進(jìn)行 HTTP 網(wǎng)絡(luò)請(qǐng)求開(kāi)發(fā)時(shí)(如使用requests ...
2025-09-15CDA 數(shù)據(jù)分析師:激活表格結(jié)構(gòu)數(shù)據(jù)價(jià)值的核心操盤(pán)手 表格結(jié)構(gòu)數(shù)據(jù)(如 Excel 表格、數(shù)據(jù)庫(kù)表)是企業(yè)最基礎(chǔ)、最核心的數(shù)據(jù)形態(tài) ...
2025-09-15Python HTTP 請(qǐng)求工具對(duì)比:urllib.request 與 requests 的核心差異與選擇指南 在 Python 處理 HTTP 請(qǐng)求(如接口調(diào)用、數(shù)據(jù)爬取 ...
2025-09-12解決 pd.read_csv 讀取長(zhǎng)浮點(diǎn)數(shù)據(jù)的科學(xué)計(jì)數(shù)法問(wèn)題 為幫助 Python 數(shù)據(jù)從業(yè)者解決pd.read_csv讀取長(zhǎng)浮點(diǎn)數(shù)據(jù)時(shí)的科學(xué)計(jì)數(shù)法問(wèn)題 ...
2025-09-12CDA 數(shù)據(jù)分析師:業(yè)務(wù)數(shù)據(jù)分析步驟的落地者與價(jià)值優(yōu)化者 業(yè)務(wù)數(shù)據(jù)分析是企業(yè)解決日常運(yùn)營(yíng)問(wèn)題、提升執(zhí)行效率的核心手段,其價(jià)值 ...
2025-09-12用 SQL 驗(yàn)證業(yè)務(wù)邏輯:從規(guī)則拆解到數(shù)據(jù)把關(guān)的實(shí)戰(zhàn)指南 在業(yè)務(wù)系統(tǒng)落地過(guò)程中,“業(yè)務(wù)邏輯” 是連接 “需求設(shè)計(jì)” 與 “用戶體驗(yàn) ...
2025-09-11塔吉特百貨孕婦營(yíng)銷案例:數(shù)據(jù)驅(qū)動(dòng)下的精準(zhǔn)零售革命與啟示 在零售行業(yè) “流量紅利見(jiàn)頂” 的當(dāng)下,精準(zhǔn)營(yíng)銷成為企業(yè)突圍的核心方 ...
2025-09-11CDA 數(shù)據(jù)分析師與戰(zhàn)略 / 業(yè)務(wù)數(shù)據(jù)分析:概念辨析與協(xié)同價(jià)值 在數(shù)據(jù)驅(qū)動(dòng)決策的體系中,“戰(zhàn)略數(shù)據(jù)分析”“業(yè)務(wù)數(shù)據(jù)分析” 是企業(yè) ...
2025-09-11Excel 數(shù)據(jù)聚類分析:從操作實(shí)踐到業(yè)務(wù)價(jià)值挖掘 在數(shù)據(jù)分析場(chǎng)景中,聚類分析作為 “無(wú)監(jiān)督分組” 的核心工具,能從雜亂數(shù)據(jù)中挖 ...
2025-09-10統(tǒng)計(jì)模型的核心目的:從數(shù)據(jù)解讀到?jīng)Q策支撐的價(jià)值導(dǎo)向 統(tǒng)計(jì)模型作為數(shù)據(jù)分析的核心工具,并非簡(jiǎn)單的 “公式堆砌”,而是圍繞特定 ...
2025-09-10