深度神经网络
摘要:深度学习是近年来计算机人工智能领域非常火的研究方向,其相比传统的浅层机器学习而言能够挖掘出更多隐含的特征。深度学习现如今已经广泛的应用与计算机视觉、自然语言处理等领域,因此作为计算机专业人工智能的学习者,学习和研究深度学习是一项必修课。
神经网络作为作为深度学习的“常客”,是成为构建深度学习框架的主要结构,神经网络以其与人类神经元相类似的元素通过相互连接形成网络拓扑结构,而这种模型能够自主挖掘更深层次特征。
建议深读的英文书籍:Neural Networks and Deep Learning
- 启发:线性回归
- 神经网络的结构
- 神经网络的训练
- 模型优化
- 其他经典的神经网络
- TensorFlow实现标准神经网络源程序
- 总结
一、启发:线性回归
神经网络的基本计算主要以线性和非线性为主,线性是为了对不同的特征进行相互组合,其主要的运算结果是线性回归问题;非线性处理是为了对模型进行优化,使其能够处理非线性问题,其主要运算以函数形式。
线性模型是机器学习中最简单的分类模型,单变量线性回归是典型的回归问题,其主要模型为:
其中 表示参数矩阵(或权重矩阵), 表示偏向向量(一元线性曲线的截距), 表示输入的特征向量, 表示对应的输出。线性分类器的目的便是训练一个线性回归方程,使得其能够拟合实际的样本输出。
通过曲线图可以描绘线性回归方程的含义:
下面给出模型训练的推导:
假设有一组训练样本,记作 {(,)|{},{}},其中 表示样本数,选择平方差作为损失函数,即:
则代价函数(所有样本的损失函数的均值)为:
和偏向,因此需要对这些参数求偏导,以获得梯度方向。
Ps:梯度下降法需要链式法则完成偏导的求解。
线性回归模型的梯度下降如下:
若选择学习率为 ,则参数调整为:
通过不断循环调参,直到参数变化非常小的时候(在编程过程中,可以设置一个变量用以判断是否需要下一轮迭代),线性回归模型可以说训练完成。在评估该模型时候,仍然可以使用损失函数来对测试集进行评估,并计算相应的准确率、召回率、F1值等。
二、神经网络的结构
深入学习神经网络后会发,神经网络便是若干个线性回归方程的“相互交织”,再通过非线性方程进行“锐化”,因此第一节的线性回归问题是解决神经网络的基础。
神经网络是由神经元组成,神经元又称信息感知机,其通过多条路径(突触)接收外界信息并传导至神经细胞,并通过激活或抑制等策略做出反应。这里的突触即为数据的输入,神经细胞即为模型的结点,而反应即为激活函数。模型图如下所示:
图(a)为一个信息感知机,输入模型的数据为 ,连接的边上的权重分别为 ,结点神经细胞获取外界的输入记为:
因此,对于单个神经元接受数据部分,是一个典型的线性回归模型。神经元的“激活或抑制反应”主要通过激活函数来完成。常用的激活函数有sigmod、tanh、ReLU、ELU等,这些激活函数都属于非线性函数,函数的输出即为该神经元对外做出的反应,即模型的输出值。以sigmod为例,该神经元的输出为:
Ps:sigmod函数的性质:
图(b)是多个神经元相互交织在一起形成网络结构,其称为深度神经网络。深度神经网络常由输入层、隐含层和输出层组成。输入层的神经元个数等于输入样本的特征属性个数,隐含层可以自定个数(通常超过3个),隐含层的数据是不可见的,输出层作为模型的输出部分,神经元个数由需要分类的类标个数决定。
三、神经网络的训练
了解了神经网络的结构,需要了解神经网络如何进行训练。神经网络的训练过程分为如下几个步骤:
- 参数设置:设置模型的参数和超参数,包括权重矩阵、偏向、梯度下降的学习率;
- 前向传播:根据模型的输入层样本数据,计算每个神经元的接受数据和输出数据,并最终求得模型的输出层数据;
- 反向传播:选择损失函数并计算代价,选择优化策略(常选择梯度下降法)并进行调参;
- 模型的评估:根据测试集计算准确率、召回率、F1值等。
假设一个深度神经网络的模型如上图所示,下面根据四步骤详解神经网络的训练过程。
1、参数设置
参数设置包括对需要学习的参数的初始化和对超参数(作为定值)的初始化。
(1)对需要学习的参数进行设置
假设神经网络输入层有三个神经元,隐含层有三个神经元,输出层有两个神经元,现在给定一组训练集,记作 ,其中 表示样本数量,,即每个样本有三个属性,分别对应神经网络的输入,每个样本对应一个真实类标(单类标问题),该类标有两个取值,设分别为0或1,即 ,神经网络中的输出层有两个输出,分别对应两种取值的概率。
神经网络每条边都有一个权值,设输入层与隐含层之间的权值矩阵为
矩阵的行数等于隐含层的神经元数,矩阵的列数等于输入层的神经元数,同理可得 :
矩阵的行数等于输出层的神经元数,矩阵的列数等于隐含层的神经元数。
在进行计算或者编程过程中,弄清各个权重矩阵的维度对于后期的检查或调试至关重要。
隐含层和输出层偏向分别记为:
(2)对超参数进行设置
神经网络在训练过程中只有梯度下降法的学习率是不需要进行学习调整的参数,通常取0.05。
2、前向传播
神经网络的前向传播主要指输入层的数据顺着边流向隐含层,经过一次非线性处理后得到的数据再一次经过边流向输出层,经过处理后得到一组输出值。前向传播主要涉及到线性运算和非线性处理,在理解上非常简单。
(1)为了详细表达运算,先以单个样本进行前向传播的详细推导
根据第一节了解了线性回归的运算,如上图,隐含层的每一个神经元都将接收三条边传来的数据,并与该神经元本身的偏向相加,即有:
根据矩阵论的知识,可知这是一组多元线性方程组,可以用矩阵方式相乘,因此可以写作:
是一个4*1的向量,将其喂给sigmod函数后,得到隐含层的输出:
其次,隐含层将数据传输给输出层,输入数据此时为
即:
最后,获得的输出喂给sigmoid函数(通常最后都喂给softmax函数,这里为了方便反向传播求导,仍然使用sigmoid函数),
因为该神经网络的输出层一共两个结点,因此对于单个样本,输出格式为[*,*]。
(2)对于多个样本,可以采用显示for循环方式按如上方式运算,但这样有明显两个缺点:一个是通过for循环显示进行循环非常耗时,第二是当数据量非常庞大时候对训练参数非常不利,因此对于多样本进行前向传播时,选择向量运算。假设样本集为 ,其含有 个样本。
(1)输入层样本:,样本维度为(3,)
(2)隐含层:,,隐含层输出值维度为(4,);
(3)输出层:,,输出层输出值维度为(2,);
Ps:在编程时,样本集每一列表示一个样本。
3、反向传播
神经网络的训练关键就是训练它的权重矩阵和偏向,反向传播就是在不断地根据当前参数对样本产生的误差进行调整,以保证该误差尽可能小。反向传播时理解神经网络算法的核心。
神经网络反向传播常用的策略是梯度下降法,梯度下降法通过偏导数的链式法则实现反向传播,选择的损失函数是交差信息熵。为了详细推导梯度下降的反向传播,同样先以单个样本为例:
(1)单样本反向传播:根据单样本的前向传播推导,得到最终的输出值:
损失函数交差信息熵表达式是:
因此可求出 对 的偏导:
由 式子可以继续求得 对 的偏导,这里便需要链式求导:
下面将要对权重矩阵和偏向进行求导,计算之前先进行必要的分析:隐层的神经元与输出层神经元相连的边都是唯一的,因此对于每一条边可通过链式法则求偏导:
以隐含层第一个神经元为例,先列出涉及到的前向传播的式子:
因此对于权重 偏导数为:
对于权重 偏导数为:
因此可以归纳出一个对于边 的偏导通向公式:
对于输出层的偏向,以输出层第一个神经元为例,列出涉及到的前向传播一个式子:
,由前向传播式子可以很简单的求出导数为:
对于隐含层的求导较为复杂,如下图,对于隐含层的某一个神经元,其将收到所有输出层神经元的反向传播,因此对于隐含层的每一个神经元的输出值求导需要分别对所有输出层进行偏导求和。
涉及到的前向传播式子还是这个:
因此可求出 对 的偏导:
由此可以列出另三个:
合并起来可以以矩阵形式呈现:
有了 对 的偏导,则可以求出 对
对于输入层与隐含层之间的权重矩阵和隐含层的偏向的求导,方法与上面的一致:
对于边 的偏导通向公式:
对于隐含层偏向 的偏导通向公式:
到此,关于神经网络的权重矩阵和偏向的梯度求解全部结束,参数的调整即为:
四、模型优化
神经网络在实际使用中,其深度不只有三层,而是大于三层,输入层。隐含层和输出层的结点也都有几十上百个神经元,因此对于一个神经网络来说,模型的参数是庞大的,因此传统的模型在一般的机器上很难快速训练参数,因此在深度神经网络的训练过程中,如何对模型进行优化成为比较现实的问题。
模型优化有几个方面:防止模型过拟合、选择其他的激活函数、选择其他的损失函数或不同的优化模型等。
1、正则化
正则化是最常用的防止过拟合的策略,其主要在代价函数后添加正则参数,如下:
其中 为L2正则化, 称为正则化参数。现对代价函数进行求导,可得:
因此权重矩阵的参数调整为:
激活函数为例,tanh激活函数的函数图像为:
可以发现,当参数 很小的时候,函数近乎为线性,而对代价函数调价正则化后,若 的值越大,则权重矩阵 的各个元素的值越小(甚至为0),因此由 可知 的值变小。所以在带入tanh激活函数中时,其近乎为线性函数,因此实现了防止过拟合。
如上图,左图属于欠拟合(高偏差),右图属于过拟合(高方差),正则化后属于中间的图。
其他正则化还有dropout,可以实现防止过拟合。
2、激活函数
激活函数包括sigmod、tanh、ReLU、ELU等,激活函数的选择在一定程度上会影响模型的训练效果。关于激活函数的详解,可参考其他内容。
3、梯度问题
由于神经网络的数据量庞大问题,除了产生过拟合,还会产生梯度衰减或梯度爆炸问题,由于梯度下降法的链式求导原则,可能会造成参数调整后过大,或趋近于0。解决梯度衰减和梯度爆炸问题,对于不同类型的网络有不同的解决方案,常用的解决方案是更换激活函数。
五、其他经典的神经网络
深度神经网络,又称为BP神经网络或者前馈神经网络,其属于标准的神经网络模型,除了这个模型外,还有其他许多类型神经网络。
1、循环神经网络RNN
循环神经网络通常可以解决序列问题。传统的BP神经网络的输入层神经元个数取决于输入样本的特征属性个数,且输入样本之间没有联系。而对于自然语言处理中相关的问题(情感分析、词性标注、命名实体识别、语音识别等)的输入数据之间往往是有关联的。例如在命名实体识别的任务中,要识别一句话“马云在浙江杭州创办了阿里巴巴”,如果单纯的对于每一个字作为神经网络的输入,模型是无法进行准确的识别,因为对于实体来说只有多个字组合起来才能形成实体,因此RNN用来解决序列问题。
关于RNN的变种还有双向循环神经网络、长短期记忆神经网络(LSTM)、双向长短期记忆神经网络(Bi-LSTM)等。
2、卷积神经网络CNN
卷积神经网络常用来进行图像处理,其通过将图像的数据((m,n,3),其中m,n为图像像素,3为三原色)进行卷积操作,再通过池化层(最大池化或平均池化)进行降维,最后通过softmax层取出最大值作为特征。卷积神经网络也可以用在文本处理中。
六、TensorFlow实现标准神经网络源程序
为了深入学习神经网络的实现过程,本人使用python的TensorFlow框架实现了BP神经网络算法,编程采用python面向对象,程序的运行流程如图:
1. main函数:实现读取训练集和测试集文件,并对数据进行处理(分为训练集X和y、测试集X和y),并调用神经网络类进行训练和预测。
#code by WangJianing
#time:2018.11.24
import tensorflow as tf
import numpy as np
from neural_network import NN
def readFile(filename):
"""
read file from txt
"""
input_x = []
input_y = []
with open(filename,'r') as f:
while True:
line = f.readline()
if line == '':
break
else:
line = line.replace('\n','')
sample = line.split(' ')
x = sample[0:3]
x = list(map(np.float32, x))
y = sample[3]
y = list(map(np.int32, y))
input_x.append(x)
input_y.append(y)
return input_x,input_y
if __name__ == '__main__':
config = tf.ConfigProto()
config.gpu_options.allow_growth = True
config.gpu_options.per_process_gpu_memory_fraction = 0.2 # need ~700MB GPU memory
train_x,train_y = readFile('./data.txt')
test_x,test_y = readFile('./data_test.txt')
sample_size = [len(train_y),len(test_y)]
print(sample_size)
train_x = np.transpose(train_x)
input_y = np.zeros([2,sample_size[0]])
test_x = np.transpose(test_x)
test_y = np.transpose(test_y)
for ei,i in enumerate(train_y):
input_y[i[0]][ei]=1
# print(ei,i)
#build neural network
n = NN(train_x, input_y, test_x, test_y, 'GradientDescentOptimizer', sample_size, config, learning_rate=0.05)
#train
n.train1()
#test
n.test()
2. 神经网络类:tensorflow创建计算图结点,神经网络参数初始化、前向传播、反向传播、最小化损失函数和梯度下降调参、测试集预测和精度评估。
#code by WangJianing
#time:2018.11.24
import tensorflow as tf
import numpy as np
class NN(object):
"""docstring for NN"""
def __init__(self, train_x, train_y, test_x, test_y, optimize, sample_size, config, learning_rate=0.05):
super(NN, self).__init__()
self.train_x = tf.to_float(train_x, name='ToFloat1')
self.train_y = tf.to_float(train_y, name='ToFloat2')
self.test_x = tf.to_float(test_x, name='ToFloat3')
self.test_y = tf.to_float(test_y, name='ToFloat4')
self.learning_rate = learning_rate
self.optimize = optimize
self.sess = tf.Session()
self.sample_size = sample_size
self.config = config
self.para = [[],[],[],[],0]
self.bildGraph()
# self.train()
def bildGraph(self):
self.parameter_op()
self.towards_op()
self.loss_op()
self.backwords_op()
# self.test_towords()
self.init_op()
def testBuildGraph(self):
self.parameter_op()
self.towards_op()
def parameter_op(self):
self.weight1 = tf.Variable(tf.random_normal([4, 3], stddev=0.03), dtype=tf.float32, name='weight1')
self.bias1 = tf.Variable(tf.random_normal([4, 1]), dtype=tf.float32, name='bias1')
self.weight2 = tf.Variable(tf.random_normal([2, 4], stddev=0.03), dtype=tf.float32, name='weight2')
self.bias2 = tf.Variable(tf.random_normal([2, 1]), dtype=tf.float32, name='bias2')
self.input_xx = tf.Variable(self.train_x,name='xx1')
self.input_xx_test = tf.Variable(self.test_x,name='xx3')
self.input_yy = tf.Variable(self.train_y,name='xx2')
def appendVector(self, v, size, kind):
#该方法是将一个一维向量v复制size次并拼起来
_v = tf.transpose(v)[0]
# print('_v=',_v)
new_v = []
if kind == 0:
for i in range(size):
new_v.append(_v)
self.bias1_train = tf.Variable(new_v, dtype=tf.float32, name='bias1_train')
self.bias1_train = tf.transpose(self.bias1_train)
elif kind == 1:
for i in range(size):
new_v.append(_v)
self.bias2_train = tf.Variable(new_v, dtype=tf.float32, name='bias2_train')
self.bias2_train = tf.transpose(self.bias2_train)
elif kind == 2:
for i in range(size):
new_v.append(_v)
self.bias1_test = tf.Variable(new_v, dtype=tf.float32, name='bias1_test')
self.bias1_test = tf.transpose(self.bias1_test)
elif kind == 3:
for i in range(size):
new_v.append(_v)
self.bias2_test = tf.Variable(new_v, dtype=tf.float32, name='bias2_test')
self.bias2_test = tf.transpose(self.bias2_test)
def towards_op(self):
self.m1 = tf.matmul(self.weight1, self.input_xx, name='matmul1')
# print('m1=',self.m1)
self.appendVector(self.bias1, self.sample_size[0], 0)
# print('self.bias1_train=',self.bias1_train)
self.z1 = tf.add(self.m1 ,self.bias1_train, name='z1')
self.a1 = tf.nn.sigmoid(self.z1,name='a1')
self.appendVector(self.bias2, self.sample_size[0], 1)
self.z2 = tf.add(tf.matmul(self.weight2, self.a1, name='matmul2'),self.bias2_train, name='z2')
self.a2 = tf.transpose(tf.nn.softmax(tf.transpose(self.z2,[1,0]),name='a2'),[1,0])
def test_towords(self):
self.t_m1 = tf.matmul(self.para[0], self.input_xx_test, name='matmul3')
self.appendVector(self.para[2], self.sample_size[1], 2)
self.t_z1 = tf.add(self.t_m1 ,self.bias1_test, name='z1')
self.t_a1 = tf.nn.sigmoid(self.t_z1,name='a1')
self.appendVector(self.para[3], self.sample_size[1], 3)
self.t_z2 = tf.add(tf.matmul(self.para[1], self.t_a1, name='matmul4'),self.bias2_test, name='z2')
self.t_a2 = tf.transpose(tf.nn.softmax(tf.transpose(self.t_z2,[1,0]),name='a2'),[1,0])
def loss_op(self):
self.loss = tf.reduce_mean(tf.nn.sigmoid_cross_entropy_with_logits(labels=self.train_y, logits=self.a2))
self.optimizer = tf.train.GradientDescentOptimizer(self.learning_rate)
def backwords_op(self):
self.train = self.optimizer.minimize(self.loss)
def train1(self):
with tf.Session(config=tf.ConfigProto(gpu_options=tf.GPUOptions(per_process_gpu_memory_fraction=0.333))) as sess:
sess.run(self.init_op)
for i in range(10):
sess.run(self.train)
self.para = [sess.run(self.weight1),sess.run(self.weight2),sess.run(self.bias1),sess.run(self.bias2),sess.run(self.loss)]
print("==========step",i,"==========")
print("weight1:\n",self.para[0],"\nb1:\n",self.para[2])
print("\nweight2:\n",self.para[1],"\nb2:\n",self.para[3])
print("\nloss=",self.para[4])
def init_op(self):
self.init_op = tf.global_variables_initializer()
def test(self):
self.test_towords()
with tf.Session(config=tf.ConfigProto(gpu_options=tf.GPUOptions(per_process_gpu_memory_fraction=0.333))) as sess:
sess.run(tf.global_variables_initializer())
sess.run([self.bias1_test,self.bias2_test])
#每个样本的每个类标取值的概率
predict_proba = sess.run(self.t_a2)
#预测每个样本的类标(0或1)
predict_proba = np.transpose(predict_proba)
print('\npredict_proba=',predict_proba)
predict_value = np.argmax(predict_proba,axis=1)
print('\npredic_value=',predict_value)
#计算准确率:
# accuracy = 0
# # print(test_y[0][0])
# for ei,i in enumerate(predict_value):
# if i == self.test_y[0][ei]:
# accuracy += 1
# accuracy /= sample_size
# print('\naccuracy=',accuracy)
七、总结
本人在编写本博文大概花了一周时间,经过对深度学习的常用模型——神经网络的原理理解、模型训练的推导以及最后程序编写,能够更加深刻的理解神经网络模型,这对今后学习机器学习、深度学习以及涉及到相关应用时具有很大的作用。
深度学习的不断发展,将越来越为生产生活提供便利,学好神经网络,是走入深度学习的必经之路,能够带着理解走入人工智能领域,对工程、科研具有巨大的意义。
本人将继续在人工智能、机器学习、深度学习领域内不断学习,不断更新博文。
博客记录着学习的脚步,分享着最新的技术,非常感谢您的阅读,本博客将不断进行更新,希望能够给您在技术上带来帮助。
标签:name,self,test,神经网络,train,深度,tf From: https://blog.51cto.com/u_15919249/5962441