@evilking
2018-05-01T18:45:58.000000Z
字数 8078
阅读 1932
机器学习篇
本篇会详细的讲解BP神经网络的原理,以及算法推导过程,BP神经网络的算法优缺点以及如何改进,最后会以一个实际的例子应用R语言编程来演示BP神经网络的过程
基本BP算法包括两个方面:信号的前向传播和误差的反向传播。即计算实际输出时按从输入到输出的方向进行,而权值和阈值的修正从输出到输入的方向反馈
假设给定训练集 ,即输入示例由 个属性描述,输出 维实值向量。
给出了一个拥有个输入神经元,个输出神经元,个隐含层神经元的多层前馈网络结构,如下图所示:
其中:
输入向量:
隐含层输入向量:
隐含层输出向量:
输出层输入向量:
输出层输出向量:
输入层第个神经元与隐含层第个神经元之间的连接权值:
隐含层第个神经元与输出层第个神经元之间的连接权值:
隐含层神经元的阈值向量:
输出层神经元的阈值向量:
样本数据个数:
激活函数: 为Sigmoid函数
如前篇所述,激活函数不只有Sigmoid函数一种,还有高斯函数、阶跃函数等等可选,只是由于Sigmoid函数性质比较好,即,所以我们一般采用Sigmoid函数作为激活函数
这里假设隐含层和输出层的激活函数都使用Sigmoid函数
给个连接权值分别赋一个区间 内的随机数 : 为了将连接权值归一化到 ,方便计算
设定误差函数 : 一般选样本的均方误差,有的选“正则化”的均方误差
给定计算精度值 : 当误差函数的值小于计算精度时,代表网络训练达到目标,可停止学习
设置最大学习次数 : 防止出现网络不收敛 (或者叫不稳定) 的情况,当迭代次数超过 则网络停止学习,但不代表网络已达到目标
随机选取训练例 ,假定神经网络的输出为
则有:
第 个隐含层神经元的输入:
第 个隐含层神经元的输出:
第 个输出层神经元的输入:
第 个输出层神经元的输出:
则网络在 上的均方误差为:
BP算法基于梯度下降(gradient descent)策略,以目标的负梯度方向对参数进行调整。因为正梯度方向是增长最快的方向,则负梯度方向就是减少最快的方向,我们要让误差函数以最快的速度减小
网络中有 个参数需要确定: 输入层到隐含层的 个权值、隐含层到输出层的 个权值、个隐含层神经元的阈值、个输出层神经元的阈值
BP是一个迭代学习算法,任意参数 的更新估计式为
具体来说:
其中: 为学习速率; 过小时,整个网络迭代的收敛速度就过慢; 过大时,网络的收敛速度就比较快,但同时容易出现震荡现象而无法收敛
因为我们要以目标的负梯度方向对参数进行调整,所以对隐含层与输出层之间的连接权值有
因为
则.
所以有
下面推导输入层与隐含层的连接权值调整公式:
注意,上面两步连接权值调整过程中的学习速率 ,可相同也可不相同
至此,连接权值调整部分的推导就算完成了,有兴趣的读者可以利用类似的方法推导其他几个参数的调整公式
[Hornik et al., 1989]证明,只需要一个包含足够多神经元的隐含层,多层前馈网络就能以任意精度逼近任意复杂度的连接函数
正是由于其强大的表示能力,BP神经网络经常遭遇过拟合,其训练误差持续降低,但测试误差却可能上升。有两种策略常用来缓解BP网络的过拟合:
第一种策略是“早停”(early stopping): 将数据分成训练集和验证集,训练集用来计算梯度、更新连接权值和阈值,验证集用来估计误差,若训练集误差降低但验证集误差升高,则停止训练,同时返回具有最小验证集误差的连接权和阈值;
第二种策略是“正则化”(regularization),其基本思想是在误差目标函数中增加一个用于描述网络复杂度的部分,例如连接权与阈值的平方和。任令 表示第 个训练样例上的误差,表示连接权和阈值,则误差目标函数改变为
BP算法因其简单、易行、计算量小、并行强等优点,目前是神经网络训练采用最多也是最成熟的训练算法之一
其算法的实质是求解误差函数的最小值问题,由于它采用非线性规划中的最速下降法,按误差函数的负梯度方向修改权值,且神经元之间采用全连接的方式,因而通常存在以下问题:
附加动量法使网络在修正其权值时,不仅考虑误差在梯度上的作用,而且考虑在误差曲面上变化趋势的影响。在没有附加动量的作用下,网络可能陷入浅的局部极小值,利用附加动量就有可能滑过这些极小值。
该方法是在反向传播法的基础上再每一个权值(或阈值)的变化上加上一项比与前次权值变化量的值,并根据反向传播法来产生新的权值变化。
带有附加动量因子的权值和阈值调节公式为:
附加动量法的实质是将最后一次权值变化的影响,通过一个动量因子来传递。当动量因子取值为 0 时,权值的变化仅是更加梯度下降法产生;当动量因子取值为 1 时,新的权值变化则是设置为最后一次权值的变化,而依梯度法产生的变化部分则被忽略掉了
以此方式,当增加了动量项后,促使权值的调节向着误差曲面底部的平均方向变化,当网络权值进入误差曲面底部的平坦区时,将变得很小,于是 ,从而防止了 的出现,有助于使网络从误差曲面的局部极小值中跳出
对于一个特定的问题,要选择适当的学习速率不是一件容易的事情。通常是凭经验或实验获取,但即使这样,对训练开始初期功效较好的学习速率,不见得对后来的训练合适。为了解决这个问题,人们自然想到在训练过程中,自动调节学习速率。
调节学习速率的准则是: 检查权值是否真正降低了误差函数,如果确实如此,则说明所选学习速率小了,可以适当增加一个量;若不是这样,而产生了过调,那么就应该减少学习速率的值。
自适应学习速率的调整公式:
当采用前述的动量法时,BP算法可以找到全局最优解,而当采用自适应学习速率时,BP算法可以缩短训练时间,采用这两种方法也可以用来训练神经网络
理论上已证明: 具有偏差和至少一个S型隐含层加上一个线性输出层的网络,能够逼近任何有理数。
增加层数可以更进一步的降低误差,提高精度,但同时也使网络复杂化,从而增加了网络权值的训练时间。
而误差精度的提高实际上也可以通过增加神经元数目来获得,其训练效果也比增加层数更容易观察和调整。
所以一般情况下,应优先考虑隐含层中的神经元数
如何设置隐含层神经元的个数任是一个未决问题,实际应用中通常靠“试错法”(trial-by-error)或者经验公式来调整。
比如
由于系统是非线性的,初始值对于学习是否达到局部最小、是否能够收敛及训练时间的长短关系很大。
如果初始值太大,使得加权后的输入和落在了 S 型激活函数的饱和区,从而导致其导数 非常小,而在计算权值修正公式中,因为 ,当时,则有。这使得,从而使得调节过程几乎停顿下来。所以一般总是希望经过初始加权后的每个神经元的输出值都接近于零,这样可以保证每个神经元的权值都能够在它们的 S型 激活函数变化最大之处进行调节。
所以,一般取初始值在之间的随机数.
或者可以先选取小量样本集进行BP神经网络的训练,将训练的结果作为全量样本训练神经网络的初值,可加快网络收敛速度,减少迭代次数.
一般情况下,倾向于选取较小的学习速率以保证系统的稳定性,学习速率一般在 0.01~0.8 之间选取。
也可参考自适应学习速率法进行设置.
library(AMORE) #加载神经网络工具包
#构造数据
x1 <- round(runif(2000,1,2000))
x2 <- round(runif(2000,1,2000))
#标准化训练集
x11 <- scale(x1[1:1900])
x12 <- scale(x2[1:1900])
#标准化验证集
x21 <- scale(x1[1901:2000])
x22 <- scale(x2[1901:2000])
#构造目标值
y1 <- x11^2 + x12^2
y2 <- x21^2 + x22^2
p <- cbind(x11, x12)
q <- cbind(x21, x22)
target = y1
#设计BP神经网络
net <- newff(n.neurons = c(2,5,1), learning.rate.global = 1e-2,momentum.global = 0.4,error.criterium = "LMS",Stao = NA, hidden.layer = "tansig",output.layer = "purelin",method = "ADAPTgdwm")
#训练神经网络
result <- train(net, p, target, error.criterium = "LMS", report = TRUE, show.step = 200, n.shows = 10)
#展示结果
z <- sim(result$net, q)
plot(q[1:100,1],z, col= "blue", pch = "+")
points(q[1:100,1],y2, col = "red", pch = "x")
训练结果如下图所示:
这里重点讲下newff()函数和train()函数,可通过help()函数查看函数详情
newff()函数可返还一个神经网络对象,它的参数介绍如下:
n.neurons
参数是一个向量,向量的第一个元素表示输入层神经元数,最后一个元素表示输出层神经元数,中间的元素个数表示隐含层的层数,每个元素的值表示对于隐含层的神经元数,如上例中n.neurons = c(2,5,1)
表示输入层有2个神经元,只有一个隐含层,隐含层的神经元数为5,输出层只有一个神经元;
learning.rate.global
参数表示每个神经元学习速率
momentum.globa
参数表示每个神经元的动量因子
error.criterium
参数表示误差函数,目前可选"LMS":最小均方误差,"LMLS":最小对数均方误差,"TAO":TAO Error
hidden.layer
参数表示隐含层神经元的激活函数,目前可选"purelin","tansig","sigmoid","hardlim","custom": 用户自定义函数
output.layer
参数表示输出层的激活函数
method
参数表示适合的学习方法
train()函数用来训练神经网络,得到各个参数的结果,它的参数介绍如下:
net
表示用来训练的神经网络对象
P
表示训练集
T
表示目标向量
error.criterium
表示评估训练结果的好坏
n.shows
表示要显示迭代结果的迭代步数
show.step
表示训练的最大迭代次数