图像预处理
使用下图(后方称为 SAMPLE_BMP)作为训练和测试数据来源,下文将讲述如何将图像转换为训练数据。
灰度化和二值化
在字符识别的过程中,识别算法不需要关心图像的彩色信息。因此,需要将彩色图像转化为灰度图像。经过灰度化处理后的图像中还包含有背景信息。因此,我们还得进一步处理,将背景噪声屏蔽掉,突显出字符轮廓信息。二值化处理就能够将其中的字符显现出来,并将背景去除掉。在一个[0,255]灰度级的灰度图像中,我们取 196 为该灰度图像的归一化值,代码如下:
- def convert_to_bw(im):
-
im = im.convert("L")
-
im.save("sample_L.bmp")
-
im = im.point(lambda x: WHITE if x > 196 else BLACK)
-
im = im.convert('1')
-
im.save("sample_1.bmp")
-
return im
def convert_to_bw(im):
im = im.convert("L")
im.save("sample_L.bmp")
im = im.point(lambda x: WHITE if x > 196 else BLACK)
im = im.convert('1')
im.save("sample_1.bmp")
return im
下图是灰度化的图像,可以看到背景仍然比较明显,有一层淡灰色:
下图是二值化的图像,可以看到背景已经完全去除:
图片的分割和规范化:
通过二值化图像,我们可以分割出每一个字符为一个单独的图片,然后再计算相应的特征值,如下图所示:
这些图片是由程序自动进行分割而成,其中用到的代码片段如下:
- def split(im):
-
assert im.mode == '1'
- result = []
- w, h = im.size
- data = im.load()
-
xs = [0, 23, 57, 77, 106, 135, 159, 179, 205, 228, w]
-
ys = [0, 22, 60, 97, 150, h]
-
for i, x in enumerate(xs):
-
if i + 1 >= len(xs):
-
break
-
for j, y in enumerate(ys):
-
if j + 1 >= len(ys):
-
break
-
box = (x, y, xs[i+1], ys[j+1])
- t = im.crop(box).copy()
-
box = box + ((i + 1) % 10, )
-
-
result.append((normalize_32_32(t, 'num_%d_%d_%d_%d_%d'%box), (i + 1) % 10))
-
return result
def split(im):
assert im.mode == '1'
result = []
w, h = im.size
data = im.load()
xs = [0, 23, 57, 77, 106, 135, 159, 179, 205, 228, w]
ys = [0, 22, 60, 97, 150, h]
for i, x in enumerate(xs):
if i + 1 >= len(xs):
break
for j, y in enumerate(ys):
if j + 1 >= len(ys):
break
box = (x, y, xs[i+1], ys[j+1])
t = im.crop(box).copy()
box = box + ((i + 1) % 10, )
# save_32_32(t, 'num_%d_%d_%d_%d_%d'%box)
result.append((normalize_32_32(t, 'num_%d_%d_%d_%d_%d'%box), (i + 1) % 10))
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 中去:
- f = open('train.data', 'wt')
-
print >>f, len(result), 256, 10
-
for input, output in result:
-
print >>f, input
-
print >>f, output
f = open('train.data', 'wt')
print >>f, len(result), 256, 10
for input, output in result:
print >>f, input
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 文件中。下面就将本文中设置和训练网络参数的程序列举如下:
- connectionRate = 1
-
learningRate = 0.008
-
desiredError = 0.001
-
maxIterations = 10000
-
iterationsBetweenReports = 100
-
inNum= 256
-
hideNum = 64
-
outNum=10
-
class NeuNet(neural_net):
-
def __init__(self):
-
neural_net.__init__(self)
-
neural_net.create_standard_array(self,(inNum, hideNum, outNum))
-
-
-
def train_on_file(self,fileName):
-
neural_net.train_on_file(self,fileName,maxIterations,iterationsBetweenReports,desiredError)
connectionRate = 1
learningRate = 0.008
desiredError = 0.001
maxIterations = 10000
iterationsBetweenReports = 100
inNum= 256
hideNum = 64
outNum=10
class NeuNet(neural_net):
def __init__(self):
neural_net.__init__(self)
neural_net.create_standard_array(self,(inNum, hideNum, outNum))
def train_on_file(self,fileName):
neural_net.train_on_file(self,fileName,maxIterations,iterationsBetweenReports,desiredError)
可以从代码中看到我们建立起一个输出层有 256 个神经元,隐藏层有 64 个神经元,输出层有 10 个神经元的ANN,其中神经层的连接率为 100%,学习率为 0.28,最大进行 10000 次迭代,并每隔 100 次报告一下学习结果。
- if __name__ == "__main__":
- ann = NeuNet()
-
ann.train_on_file("train.data")
-
ann.save("number_char_recognize2.net")
if __name__ == "__main__":
ann = NeuNet()
ann.train_on_file("train.data")
ann.save("number_char_recognize2.net")
按照上面的程序,对网络进行训练和仿真测试,保存训练性能最好的一组网络权值,并保存到起来。
通过 666 次迭代之后,错误率已经低于 0.001,学习中止,并将结果保存起来。
测试结果
实验的测试是通过从保存好的 NN 数据文件中创建 NN 的形式来实验的,具体的代码如下:
- if __name__ == "__main__":
- ann = NeuNet()
-
ann.create_from_file("number_char_recognize.net")
- data = read_test_data()
-
for k, v in data.iteritems():
- k = string_to_list(k)
- v = string_to_list(v)
- result = ann.run(k)
-
print euclidean_distance(v, result)
if __name__ == "__main__":
ann = NeuNet()
ann.create_from_file("number_char_recognize.net")
data = read_test_data()
for k, v in data.iteritems():
k = string_to_list(k)
v = string_to_list(v)
result = ann.run(k)
print euclidean_distance(v, result)
其实 ann.create_from_file 是从文件中读取存档,创建人工神经网络,然后使用 read_test_data 函数读取测试数据,并通过循环对每一个测试数据和相应的期望值转换为 NN 的输入格式,然后使用 ann.run 函数调用神经网络测试,对测试结果与期望值进行欧氏距离计算,对其中的两个测试用例,果如下:
可见两个向量的欧氏距离已经接近于 0,识别效果非常好
分享到:
相关推荐
在这个场景中,我们将重点讨论如何使用BP神经网络来实现手写数字识别,特别是在MATLAB环境中。 手写数字识别是模式识别领域的一个经典问题,主要目标是通过计算机程序自动识别手写数字。这项技术在许多实际应用中都...
在IT领域,手写数字识别是一项重要的应用技术,广泛应用于移动设备、银行支票自动处理等领域。本项目涉及的核心技术是“贝叶斯最小错误率”(Bayesian Minimum Risk)方法,这是一种基于概率理论的决策策略,用于...
手写数字识别是计算机视觉领域的一个重要应用,它利用机器学习和图像处理技术来自动识别手写的数字。在本项目中,我们使用MATLAB作为主要的编程环境,结合神经网络来实现这一功能。MATLAB是一种强大的数值计算和编程...
手写数字识别 手写数字识别是模式识别领域中最成功的应用之一,研究热点之一。随着计算机技术和数字图像处理技术的飞速发展,手写数字识别在电子商务、机器自动输入等场合已经获得成功的实际应用。但是,机器的识别...
标题中的"SVM手写数字识别"指的是支持向量机(Support Vector Machine,简称SVM)在手写数字识别领域的应用。SVM是一种监督学习模型,主要用于分类和回归分析,尤其在小样本数据集上表现优秀。手写数字识别是模式...
手写数字识别是一种计算机视觉领域的关键技术,主要用于读取和理解人类手写的数字,例如在银行支票、邮政编码或各种表单中的应用。这个领域的研究始于20世纪80年代,随着深度学习技术的发展,其准确性和实用性得到了...
1、资源内容:基于SVM的手写数字图像识别 2、使用/学习目标:了解手写数字图像识别 3、应用场景:基于Minist数据集的手写数字图像处理等内容实现 4、特点:基于SVM手写数字图像识别代码实现过程 5、适用人群:想阅读手写...
在计算机视觉领域,手写数字识别是一项常见的任务,它通常涉及到图像处理、机器学习和模式识别等技术。在这个项目中,我们将重点讨论如何使用C++编程语言来实现这一功能。C++由于其高效性和灵活性,成为了开发这类...
**VC手写数字识别系统**是一种利用计算机视觉和模式识别技术来识别用户手写数字的应用。这个系统通常基于机器学习算法,尤其是深度学习中的卷积神经网络(CNN),旨在模仿人类大脑对图像的理解过程,从而实现对数字...
手写数字的识别可以分成两大板块:一、手写数字模型的训练;二、手写数字的识别。其中最为关键的环节是手写数字模型的训练。本次选取使用的模型是多元线性回归模型。手写数字有 10 中,分别是 0~9,所以可以将该问题...
《基于人工神经网络的MATLAB手写数字识别系统详解》 在信息技术日益发达的今天,计算机视觉技术在各个领域都发挥着重要的作用。手写数字的自动识别是计算机视觉的一个重要应用,它广泛应用于银行支票自动处理、邮政...
深度学习基于Matlab神经网络的手写数字识别系统源代码。在一张图像上面手写了很多手写数字。利用鼠标进行框定你所要识别的数字区域。裁剪灰度化处理,二值化处理。提取数字特征。利用神经网络的方法进行识别。带有...
这个压缩包"手写体数字识别界面程序.rar"包含了一个实现手写数字识别的简易系统,它利用了模糊识别和概率理论,特别是贝叶斯定理。以下是关于这个主题的详细知识: 1. **模糊模式识别**:模糊模式识别是一种处理不...
手写数字识别是一种计算机视觉技术,它通过模拟人类大脑对图像的理解能力来识别手写的数字。在本项目中,我们采用MATLAB作为编程环境,利用机器学习算法来实现这一功能。MATLAB是一款强大的数学计算软件,它提供了...
基于 BP 神经网络的手写数字识别实验报告.pdf 本文结合深度学习理论,利用 BP 神经网络对手写体数字数据集 MNIST 进行分析,作为机器学习课程的一次实践。主要介绍了 BP 神经网络在手写数字识别中的应用,并对神经...
基于LeNet5的手写数字识别系统源码基于LeNet5的手写数字识别系统源码。使用说明 train.py 训练 test.py 测试 Mnist 手写数字数据集 readMnist.py 里配置 Mnist 路径( fpath ) test.py 配置训练的模型 包含了 lenet...
标题中的“【手写数字识别】基于RBM神经网络手写数字识别含Matlab源码.zip”揭示了这个压缩包内容的核心,它涉及到的是一个手写数字识别的项目,使用了Restricted Boltzmann Machines(RBM)神经网络,并且提供了...
本项目聚焦于一个特定的应用:手写数字识别,这通常涉及到图像预处理、特征提取和分类算法。下面将详细阐述相关知识点。 一、OpenCV简介 OpenCV是一个跨平台的计算机视觉库,由一系列C++函数和类组成,同时也提供了...