图片验证码识别算法
最后更新于:2022-04-01 06:37:11
以DZ论坛验证码识别为例:
对于比较复杂的验证码,比如DZ论坛最新的验证码,处理起来相对麻烦一些,但是原理还是和普通的识别一样的,无非多了个背景处理的方案,看如下对DZ论坛的验证码的识别的思路。
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2015-12-30_56837dede5502.png)
首先我们要去除它的背景,对于这样稍微复杂的背景,用过去的方法很难做到,上图的例子还不是很明显,我发现很多图片背景色和字母色近似,而且字母颜色是不断变化的,背景也是不断变化的。
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2015-12-30_56837dedf2d2f.png)
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2015-12-30_56837dee0af93.png)
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2015-12-30_56837dee173b8.png)
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2015-12-30_56837dee283f1.png)
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2015-12-30_56837dee3670a.png)
那我初始的想法是找到图片中使用颜色最多的方法,于是我们用HSL表示各点颜色,接着进行统计,得到最大的几个峰值,这里便是图片中几个最丰富的颜色的L值得累加值。
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2015-12-30_56837dee43443.png)
其余的都可以认为是噪音,我们对每个峰值进行分割,得到如下图片
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2015-12-30_56837dee4eff7.png)
你看这样就把单个颜色图片分割出来了,接下来就是找到图片中除去黑色和白色后的图片:
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2015-12-30_56837dee609de.png)
然后进行灰化处理,阀值处理,降噪,得到:
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2015-12-30_56837dee6ea2d.png)
接着根据边界检测出来的最左侧x位置,来排序字母顺序:
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2015-12-30_56837dee7c8a0.png)
接下来的事情就轻车熟路了,把图片转成标准模板,通过少量学习就达到了95%以上的识别率:
c:15 j:8 8:7 t:9 9:4 x:7 4:6 2:4 h:7 f:8 e:18 b:5 y:3 k:4 w:3 g:5 3:5 7:6 r:2 m:3 q:4 v:2 p:3 6:2
以上数据表示 c学习15次 j学习8次…
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2015-12-30_56837dee8b823.png)
只要字符不粘连,大部分验证码干扰技术都是可以有办法,所以为什么google验证码看起来很简单,但是没有人能够很好的破解它得原因。
补充,
有一些字符加入杂点的问题,由于这种验证码不是很普遍,稍微做了研究
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2015-12-30_56837deea6f21.png)
CY3E 这个图片3字中有杂点,其他没有,按照文章中介绍的办法,怎么知道这个3不是像其他颜色杂点一样的图片呢?
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2015-12-30_56837deeb3bf4.png)
我觉得需要加入一个步骤,就是对每次过滤颜色生成出来的图片,进行填充
找到3的杂点原图:
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2015-12-30_56837deeca921.png)
然后我们进行[算法](http://cpro.baidu.com/cpro/ui/uijs.php?adclass=0&app_id=0&c=news&cf=1001&ch=0&di=128&fv=18&is_app=0&jk=ca709a7c2cee8241&k=%CB%E3%B7%A8&k0=%CB%E3%B7%A8&kdi0=0&luki=1&n=10&p=baidu&q=67051059_cpr&rb=0&rs=1&seller_id=1&sid=4182ee2c7c9a70ca&ssp2=1&stid=0&t=tpclicked3_hc&td=1740074&tu=u1740074&u=http%3A%2F%2Fwww%2Eeducity%2Ecn%2Flabs%2F649007%2Ehtml&urlid=0)填充:
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2015-12-30_56837deed7ed5.png)
这个图片与其他全部是杂点的图片之间的差别进行过滤,我考虑可以通过以下方法:
1、连贯点的宽度
2、连贯点的个数
这样剩下的就只剩下CY3E的过滤后的图片了。
至于字符倾斜的问题,我觉得完全可以在[机器学习](http://cpro.baidu.com/cpro/ui/uijs.php?adclass=0&app_id=0&c=news&cf=1001&ch=0&di=128&fv=18&is_app=0&jk=ca709a7c2cee8241&k=%BB%FA%C6%F7%D1%A7%CF%B0&k0=%BB%FA%C6%F7%D1%A7%CF%B0&kdi0=0&luki=2&n=10&p=baidu&q=67051059_cpr&rb=0&rs=1&seller_id=1&sid=4182ee2c7c9a70ca&ssp2=1&stid=0&t=tpclicked3_hc&td=1740074&tu=u1740074&u=http%3A%2F%2Fwww%2Eeducity%2Ecn%2Flabs%2F649007%2Ehtml&urlid=0)过程中,我们自己旋转正在学习的图片一定角度,例如从-10到+10度,只不过这样的学习库会大一些,但是就10个数字的验证码来说,这点性能损失应该可以忽略不计。