赖勇浩(http://laiyonghao.com)

这是我读工程硕士的时候完成课程作业时做的,放在 dropbox 的角落中生尘已经有若干年头了,最近 @shugelee 同学突然来了兴致搞验证码识别,问到我的时候我记起自己做过一点点东西,特发上来给他参考,并趁机补充了一下《Python也可以》系列。

图像预处理

使用下图(后方称为 SAMPLE_BMP)作为训练和测试数据来源,下文将讲述如何将图像转换为训练数据。

灰度化和二值化

在字符识别的过程中,识别算法不需要关心图像的彩色信息。因此,需要将彩色图像转化为灰度图像。经过灰度化处理后的图像中还包含有背景信息。因此,我们还得进一步处理,将背景噪声屏蔽掉,突显出字符轮廓信息。二值化处理就能够将其中的字符显现出来,并将背景去除掉。在一个[0,255]灰度级的灰度图像中,我们取 196 为该灰度图像的归一化值,代码如下:

[python] view plain copy
  1. def convert_to_bw(im):
  2. im = im.convert("L")
  3. im.save("sample_L.bmp")
  4. im = im.point(lambda x: WHITE if x > 196 else BLACK)
  5. im = im.convert('1')
  6. im.save("sample_1.bmp")
  7. return im

下图是灰度化的图像,可以看到背景仍然比较明显,有一层淡灰色:

下图是二值化的图像,可以看到背景已经完全去除:

图片的分割和规范化:

通过二值化图像,我们可以分割出每一个字符为一个单独的图片,然后再计算相应的特征值,如下图所示:

这些图片是由程序自动进行分割而成,其中用到的代码片段如下:

[python] view plain copy
  1. def split(im):
  2. assert im.mode == '1'
  3. result = []
  4. w, h = im.size
  5. data = im.load()
  6. xs = [0, 23, 57, 77, 106, 135, 159, 179, 205, 228, w]
  7. ys = [0, 22, 60, 97, 150, h]
  8. for i, x in enumerate(xs):
  9. if i + 1 >= len(xs):
  10. break
  11. for j, y in enumerate(ys):
  12. if j + 1 >= len(ys):
  13. break
  14. box = (x, y, xs[i+1], ys[j+1])
  15. t = im.crop(box).copy()
  16. box = box + ((i + 1) % 10, )
  17. #           save_32_32(t, 'num_%d_%d_%d_%d_%d'%box)
  18. result.append((normalize_32_32(t, 'num_%d_%d_%d_%d_%d'%box), (i + 1) % 10))
  19. return result

其中的 xs 和 ys 分别是横向和竖向切割的分界点,由手工测试后指定,t = im.crop(box).copy() 代码行是从指定的区域中“抠”出图片,然后通过 normalize_32_32 进行规范化。进行规范化是为了产生规则的训练和测试数据集,也是为了更容易地地计算出特征码。

产生训练数据集和测试数据集

为简单起见,我们使用了最简单的图像特征——黑色像素在图像中的分布来进行训练和测试。首先,我们把图像规范化为 32*32 像素的图片,然后按 2*2 分切成 16*16 共 256 个子区域,然后统计这 4 个像素中黑色像素的个数,组成 256 维的特征矢量,如下是数字 2 的一个特征矢量:
0 0 4 4 4 2 0 0 0 0 0 0 0 0 2 4 0 0 4 4 4 2 0 0 0 0 0 0 0 0 2 4 2 2 4 4 2 1 0 0 0 0 0 0 1 2 3 4 4 4 4 4 0 0 0 0 0 0 0 0 2 4 4 4 4 4 4 4 0 0 0 0 0 0 0 0 2 4 4 4 4 4 0 0 0 0 0 0 0 0 0 0 2 4 4 4 4 4 0 0 0 0 0 0 0 0 0 0 2 4 4 4 4 4 0 0 0 0 0 0 0 4 4 4 4 4 4 4 4 4 0 0 0 0 0 0 0 4 4 4 4 4 4 4 4 4 2 2 2 2 2 2 2 4 4 2 3 4 4 4 4 4 4 4 4 4 4 4 4 4 4 0 2 4 4 4 2 2 2 2 4 3 2 2 2 2 2 0 2 4 4 4 0 0 0 0 4 2 0 0 0 0 0 0 2 4 4 4 0 0 0 0 4 2 0 0 0 0 0 0 2 4 4 4 0 0 0 0 0 0 0 0 0 0 0 0 2 4 4 4 0 0 0 0 0 0 0 0 0 0 0 0 2 4 4 4
相应地,因为我们只需要识别 0~9 共 10 个数字,所以创建一个 10 维的矢量作为结果,数字相应的维置为 1 值,其它值为 0。数字 2 的结果如下:0 0 1 0 0 0 0 0 0 0
我们特征矢量和结果矢量通过以下代码计算出来后,按 FANN 的格式把它们存到 train.data 中去:

[python] view plain copy
  1. f = open('train.data', 'wt')
  2. print >>f, len(result), 256, 10
  3. for input, output in result:
  4. print >>f, input
  5. print >>f, output

BP神经网络

利用神经网络识别字符是本文的另外一个关键阶段,良好的网络性能是识别结果可靠性的重要保证。这里就介绍如何利用BP 神经网络来识别字符。反向传播网络(即:Back-Propagation Networks ,简称:BP 网络)是对非线性可微分函数进行权值训练的多层前向网络。在人工神经网络的实际应用中,80%~90%的模型采用 BP 网络。它主要用在函数逼近,模式识别,分类,数据压缩等几个方面,体现了人工神经网络的核心部分。

网络结构

网络结构的设计是根据输入结点和输出结点的个数和网络性能来决定的,如下图。本实验中的标准待识别字符的大小为 32*32 的二值图像,即将 1024 个像素点的图像转化为一个 256 维的列向量作为输入。由于本实验要识别出10 个字符,可以将目标输出的值设定为一个10 维的列向量,其中与字符相对应那个位为1,其他的全为0 。根据实际经验和试验确定,本文中的网络隐含层结点数目为64。因此,本文中的BP 网络的结构为 256-64-10。

训练结果

本实验中的采用的样本个数为 50 个,将样本图像进行预处理,得到处理后的样本向量P,再设定好对应的网络输出目标向量T,把样本向量 P 和网络输出目标向量 T 都保存到 train.data 文件中。设置好网络训练参数,对网络进行训练和测试,并将最佳的一个网络权值保存到 number_char_recognize.net 文件中。下面就将本文中设置和训练网络参数的程序列举如下:

[python] view plain copy
  1. connectionRate = 1
  2. learningRate = 0.008
  3. desiredError = 0.001
  4. maxIterations = 10000
  5. iterationsBetweenReports = 100
  6. inNum= 256
  7. hideNum = 64
  8. outNum=10
  9. class NeuNet(neural_net):
  10. def __init__(self):
  11. neural_net.__init__(self)
  12. neural_net.create_standard_array(self,(inNum, hideNum, outNum))
  13. def train_on_file(self,fileName):
  14. neural_net.train_on_file(self,fileName,maxIterations,iterationsBetweenReports,desiredError)

可以从代码中看到我们建立起一个输出层有 256 个神经元,隐藏层有 64 个神经元,输出层有 10 个神经元的ANN,其中神经层的连接率为 100%,学习率为 0.28,最大进行 10000 次迭代,并每隔 100 次报告一下学习结果。

[python] view plain copy
  1. if __name__ == "__main__":
  2. ann = NeuNet()
  3. ann.train_on_file("train.data")
  4. ann.save("number_char_recognize2.net")

按照上面的程序,对网络进行训练和仿真测试,保存训练性能最好的一组网络权值,并保存到起来。

通过 666 次迭代之后,错误率已经低于 0.001,学习中止,并将结果保存起来。

测试结果

实验的测试是通过从保存好的 NN 数据文件中创建 NN 的形式来实验的,具体的代码如下:

[python] view plain copy
  1. if __name__ == "__main__":
  2. ann = NeuNet()
  3. ann.create_from_file("number_char_recognize.net")
  4. data = read_test_data()
  5. for k, v in data.iteritems():
  6. k = string_to_list(k)
  7. v = string_to_list(v)
  8. result = ann.run(k)
  9. print euclidean_distance(v, result)

其实 ann.create_from_file 是从文件中读取存档,创建人工神经网络,然后使用 read_test_data 函数读取测试数据,并通过循环对每一个测试数据和相应的期望值转换为 NN 的输入格式,然后使用 ann.run 函数调用神经网络测试,对测试结果与期望值进行欧氏距离计算,对其中的两个测试用例,果如下:

可见两个向量的欧氏距离已经接近于 0,识别效果非常好。

小结

本文为该项研究的初步实验阶段,由于样本字符的数目较少,选取了50 个样本用来训练,对10 个待检数字字符进行识别和仿真,成功识别出字符的个数为9 个,识别效率为90.0%。对于神经网络而言,在这样少的训练样本的情况下,能够取的这种效果已经比较成功,表明该方法具有较好识别性能。

用BP人工神经网络识别手写数字——《Python也可以》之三相关推荐

  1. 利用python实现简单的人工神经网络识别手写数字

    利用 Python 搭建起了一个简单的神经网络模型,并完成识别手写数字. 1.前置工作 1.1 环境配置 这里使用scikit-learn库内建的手写数字字符集作为本文的数据集.scikit-lear ...

  2. Python神经网络识别手写数字-MNIST数据集

    Python神经网络识别手写数字-MNIST数据集 一.手写数字集-MNIST 二.数据预处理 输入数据处理 输出数据处理 三.神经网络的结构选择 四.训练网络 测试网络 测试正确率的函数 五.完整的 ...

  3. 四、用简单神经网络识别手写数字(内含代码详解及订正)

    本博客主要内容为图书<神经网络与深度学习>和National Taiwan University (NTU)林轩田老师的<Machine Learning>的学习笔记,因此在全 ...

  4. python识别数字程序_python实现识别手写数字 python图像识别算法

    写在前面 这一段的内容可以说是最难的一部分之一了,因为是识别图像,所以涉及到的算法会相比之前的来说比较困难,所以我尽量会讲得清楚一点. 而且因为在编写的过程中,把前面的一些逻辑也修改了一些,将其变得更 ...

  5. BP神经网络识别手写数字项目解析及代码

    这两天在学习人工神经网络,用传统神经网络结构做了一个识别手写数字的小项目作为练手.点滴收获与思考,想跟大家分享一下,欢迎指教,共同进步. 平常说的BP神经网络指传统的人工神经网络,相比于卷积神经网络( ...

  6. 第1章使用神经网络识别手写数字

    人类视觉系统是世界奇观之一.考虑以下手写数字序列: 大多数人毫不费力地将这些数字识别为504192.这很容易就是欺骗性的.在我们大脑的每个半球,人类有一个主要的视觉皮层,也被称为V1,包含1.4亿个神 ...

  7. python手机代码识别数字_利用python构建神经网络识别手写数字(附源代码)

    一.运行环境配置 本次实验的运行环境win10(bit64),采用python环境为3.7.6,安装Python环境推荐使用Anaconda.Anaconda是一个免费开源的Python和R语言的发行 ...

  8. python手写数字识别实验报告_python实现识别手写数字 python图像识别算法

    写在前面 这一段的内容可以说是最难的一部分之一了,因为是识别图像,所以涉及到的算法会相比之前的来说比较困难,所以我尽量会讲得清楚一点. 而且因为在编写的过程中,把前面的一些逻辑也修改了一些,将其变得更 ...

  9. BP神经网络识别手写数字项目解析及matlab实现

    BP神经网络指传统的人工神经网络,相比于卷积神经网络(CNN)来说要简单些. 人工神经网络具有复杂模式和进行联想.推理记忆的功能, 它是解决某些传统方法所无法解决的问题的有力工具.目前, 它日益受到重 ...

最新文章

  1. oracle转金额,ORACLE金额转换成英文大写的函数
  2. python游戏编程入门书籍推荐-游戏编程入门书籍推荐:想要游戏编程尽快入门这些书不要错过...
  3. eventfd和timerfd
  4. php make,安装PHP出现make:
  5. 大型项目使用Automake/Autoconf完成编译配置
  6. angularjs mysql_AngularJS SQL
  7. Code First Migrations更新数据库结构的具体步骤
  8. 使用Dapper处理多个结果集和多重映射的教程
  9. log4j deadlock
  10. 百度要革自己的命?移动搜索或取消PC网页收录
  11. Android 应用程序模块: 应用, 任务, 进程, 和线程
  12. 红帽 linux 安装gns3,Linux下安装GNS3
  13. linux ac97声卡驱动下载,AC97声卡如何在Linux操作系统中进行驱动
  14. C# 阿里云视频点播
  15. html内嵌式选择器,CSS样式 CSS选择器(Cascading Style Sheet)
  16. Object.freeze()
  17. 单词数(HDU 2072)
  18. Power BI 矩阵总计放表第一列
  19. 网易云歌单歌曲实时拉取
  20. 【强化学习】双深度Q网络(DDQN)求解倒立摆问题 + Pytorch代码实战

热门文章

  1. 马云:蚂蚁金服这样做区块链!
  2. 程序员如果也能像C罗一样自律和勤奋,必将成为大神!
  3. 阿里巴巴的五大平台野心,让“连接”论成为过去式
  4. C语言文件读写(2)-文本文件写操作
  5. jvm性能调优实战 - 48无限循环调用和没有缓存的动态代理引起的OOM
  6. MySQL-性能优化_影响MySQL性能的因素分析及解决方案
  7. 学习笔记Hive(四) —— Hive应用(2)—— Hive导入及导出数据
  8. Spring data jpa 条件查询-按时间段查询
  9. 解决 org.hibernate.HibernateException: No Hibernate Session bound to thread, and configuration does no
  10. 怎样查看was的服务器信息,WAS 查看服务状态