python bp神经网络代码实现预测,用Python实现BP神经网络(附代码)

用Python实现出来的机器学习算法都是什么样子呢? 前两期线性回归及逻辑回归项目已发布(见文末链接),今天来讲讲BP神经网络。

BP神经网络

全部代码

https://github.com/lawlite19/MachineLearning_Python/blob/master/NeuralNetwok/NeuralNetwork.py

神经网络model

先介绍个三层的神经网络,如下图所示

输入层(input layer)有三个units(

adb64ef01a4aaea9a265e7973178c4b7.png

为补上的bias,通常设为1)

8735589089a6fa795b0dae48aff8031a.png

表示第j层的第i个激励,也称为为单元unit

ee17e2711e9150b7341ebfb52f5490dd.png

为第j层到第j+1层映射的权重矩阵,就是每条边的权重

python bp神经网络代码实现预测,用Python实现BP神经网络(附代码)_第1张图片

所以可以得到:

隐含层:

5e66c248fcc126952e2bddef5a98a658.png

f5a5a99878adc37162af89f6ffff3582.png

b6419fea71e58d569121b1a821b7fa96.png

输出层

72102a2a7282ec49fd51e7407e40f024.png

其中,S型函数

88e4fb88acb963899e7b33e8550f5d1f.png

,也成为激励函数

可以看出

68b01e070903f559b9ed847e1f50a76f.png

为3x4的矩阵,

5975cf091c125fd7dcaa40c6625b9c05.png

为1x4的矩阵

ee17e2711e9150b7341ebfb52f5490dd.png

==》j+1的单元数x(j层的单元数+1)

代价函数

假设最后输出的

82477b7fb165d742759a4f05c0e1ceed.png

,即代表输出层有K个单元

b1a6676407d8d3c11bedf41eddc9ff00.png

其中,

2b2e4e2d82cee04097d65d1a0bd941ac.png

代表第i个单元输出与逻辑回归的代价函数

6d0341bc93796a83744d7aabb16f44a7.png

差不多,就是累加上每个输出(共有K个输出)

正则化

L-->所有层的个数

a97db092ad004c002fb570de479ee391.png

-->第l层unit的个数

正则化后的代价函数为

f564cd32c6fdabdc15a4f55ad052e8c7.png

dbf0a32c57e58ca16b0f9f031b2570f7.png

共有L-1层,然后是累加对应每一层的theta矩阵,注意不包含加上偏置项对应的theta(0)

正则化后的代价函数实现代码:

# 代价函数

def nnCostFunction(nn_params,input_layer_size,hidden_layer_size,num_labels,X,y,Lambda):

length = nn_params.shape[0] # theta的中长度

# 还原theta1和theta2

Theta1 = nn_params[0:hidden_layer_size*(input_layer_size+1)].reshape(hidden_layer_size,input_layer_size+1)

Theta2 = nn_params[hidden_layer_size*(input_layer_size+1):length].reshape(num_labels,hidden_layer_size+1)

# np.savetxt("Theta1.csv",Theta1,delimiter=',')

m = X.shape[0]

class_y = np.zeros((m,num_labels)) # 数据的y对应0-9,需要映射为0/1的关系

# 映射y

for i in range(num_labels):

class_y[:,i] = np.int32(y==i).reshape(1,-1) # 注意reshape(1,-1)才可以赋值

'''去掉theta1和theta2的第一列,因为正则化时从1开始'''

Theta1_colCount = Theta1.shape[1]

Theta1_x = Theta1[:,1:Theta1_colCount]

Theta2_colCount = Theta2.shape[1]

Theta2_x = Theta2[:,1:Theta2_colCount]

# 正则化向theta^2

term = np.dot(np.transpose(np.vstack((Theta1_x.reshape(-1,1),Theta2_x.reshape(-1,1)))),np.vstack((Theta1_x.reshape(-1,1),Theta2_x.reshape(-1,1))))

'''正向传播,每次需要补上一列1的偏置bias'''

a1 = np.hstack((np.ones((m,1)),X))

z2 = np.dot(a1,np.transpose(Theta1))

a2 = sigmoid(z2)

a2 = np.hstack((np.ones((m,1)),a2))

z3 = np.dot(a2,np.transpose(Theta2))

h = sigmoid(z3)

'''代价'''

J = -(np.dot(np.transpose(class_y.reshape(-1,1)),np.log(h.reshape(-1,1)))+np.dot(np.transpose(1-class_y.reshape(-1,1)),np.log(1-h.reshape(-1,1)))-Lambda*term/2)/m

return np.ravel(J)

反向传播BP

上面正向传播可以计算得到J(θ),使用梯度下降法还需要求它的梯度

BP反向传播的目的就是求代价函数的梯度

假设4层的神经网络,

5c61169768411928fe3b77b5ac61da9c.png

记为-->l层第j个单元的误差

62279434cfc8254c8ac4c94ac50f978c.png

《===》

c32b319d5128b8a2a5ed6bd330ed815f.png

(向量化)

ed7f488fd5b28a61a56aeaf0e13e3542.png

c0e9b38afb3330cd4b46df869b590784.png

没有

358daf973b396402e5b35af32811b974.png

,因为对于输入没有误差

因为S型函数

295f6a431cbd086a23b4492a14dca960.png

的倒数为:

97cd5306cd356a7b8acbc5cdc83a92b8.png

所以上面的

b0a172ddb7439a4f0fb9c00a761d42bb.png

c0b84f38d7dd43fc3112f921e39da0cd.png

可以在前向传播中计算出来

反向传播计算梯度的过程为:

565607e96de1d92ac9391d274492d6e6.png

(

7ef32a4c7e2b23d7dd3e4d29cf9d6cbd.png

是大写的

8177b6d5e110f93fafc98580ffe00e89.png

)

for i=1-m:-

52942a3498bd92d011745519dfe5affb.png

-正向传播计算

29affa8883feb0964472c1d3e92e97b4.png

(l=2,3,4...L)

-反向计算

76e97190a246c18d7b5c8a3be126591a.png

83d318c1c5bcf9e854a2a08503794c6a.png

...

f50af640c913bec295d0f20a0fc7d85e.png

-

bc6a04fede0a1a9dddebd7a5286a51c1.png

-

8ddf589e3a8f585d6ea45a1b09847e1f.png

9de366abc4b3c166f641f5975fcf5441.png

最后

ebfcd3de0323a3edec2be3707d89feb4.png

,即得到代价函数的梯度

实现代码:

# 梯度

def nnGradient(nn_params,input_layer_size,hidden_layer_size,num_labels,X,y,Lambda):

length = nn_params.shape[0]

Theta1 = nn_params[0:hidden_layer_size*(input_layer_size+1)].reshape(hidden_layer_size,input_layer_size+1)

Theta2 = nn_params[hidden_layer_size*(input_layer_size+1):length].reshape(num_labels,hidden_layer_size+1)

m = X.shape[0]

class_y = np.zeros((m,num_labels)) # 数据的y对应0-9,需要映射为0/1的关系

# 映射y

for i in range(num_labels):

class_y[:,i] = np.int32(y==i).reshape(1,-1) # 注意reshape(1,-1)才可以赋值

'''去掉theta1和theta2的第一列,因为正则化时从1开始'''

Theta1_colCount = Theta1.shape[1]

Theta1_x = Theta1[:,1:Theta1_colCount]

Theta2_colCount = Theta2.shape[1]

Theta2_x = Theta2[:,1:Theta2_colCount]

Theta1_grad = np.zeros((Theta1.shape)) #第一层到第二层的权重

Theta2_grad = np.zeros((Theta2.shape)) #第二层到第三层的权重

Theta1[:,0] = 0;

Theta2[:,0] = 0;

'''正向传播,每次需要补上一列1的偏置bias'''

a1 = np.hstack((np.ones((m,1)),X))

z2 = np.dot(a1,np.transpose(Theta1))

a2 = sigmoid(z2)

a2 = np.hstack((np.ones((m,1)),a2))

z3 = np.dot(a2,np.transpose(Theta2))

h = sigmoid(z3)

'''反向传播,delta为误差,'''

delta3 = np.zeros((m,num_labels))

delta2 = np.zeros((m,hidden_layer_size))

for i in range(m):

delta3[i,:] = h[i,:]-class_y[i,:]

Theta2_grad = Theta2_grad+np.dot(np.transpose(delta3[i,:].reshape(1,-1)),a2[i,:].reshape(1,-1))

delta2[i,:] = np.dot(delta3[i,:].reshape(1,-1),Theta2_x)*sigmoidGradient(z2[i,:])

Theta1_grad = Theta1_grad+np.dot(np.transpose(delta2[i,:].reshape(1,-1)),a1[i,:].reshape(1,-1))

'''梯度'''

grad = (np.vstack((Theta1_grad.reshape(-1,1),Theta2_grad.reshape(-1,1)))+Lambda*np.vstack((Theta1.reshape(-1,1),Theta2.reshape(-1,1))))/m

return np.ravel(grad)

BP可以求梯度的原因

实际是利用了链式求导法则

因为下一层的单元利用上一层的单元作为输入进行计算

大体的推导过程如下,最终我们是想预测函数与已知的y非常接近,求均方差的梯度沿着此梯度方向可使代价函数最小化。可对照上面求梯度的过程。

python bp神经网络代码实现预测,用Python实现BP神经网络(附代码)_第2张图片

求误差更详细的推导过程:

python bp神经网络代码实现预测,用Python实现BP神经网络(附代码)_第3张图片

梯度检查

检查利用BP求的梯度是否正确

利用导数的定义验证:

5bd88fb25b7bba2e15a78c7888232f9f.png

求出来的数值梯度应该与BP求出的梯度非常接近

验证BP正确后就不需要再执行验证梯度的算法了

实现代码:

# 检验梯度是否计算正确

# 检验梯度是否计算正确

def checkGradient(Lambda = 0):

'''构造一个小型的神经网络验证,因为数值法计算梯度很浪费时间,而且验证正确后之后就不再需要验证了'''

input_layer_size = 3

hidden_layer_size = 5

num_labels = 3

m = 5

initial_Theta1 = debugInitializeWeights(input_layer_size,hidden_layer_size);

initial_Theta2 = debugInitializeWeights(hidden_layer_size,num_labels)

X = debugInitializeWeights(input_layer_size-1,m)

y = 1+np.transpose(np.mod(np.arange(1,m+1), num_labels))# 初始化y

y = y.reshape(-1,1)

nn_params = np.vstack((initial_Theta1.reshape(-1,1),initial_Theta2.reshape(-1,1))) #展开theta

'''BP求出梯度'''

grad = nnGradient(nn_params, input_layer_size, hidden_layer_size,

num_labels, X, y, Lambda)

'''使用数值法计算梯度'''

num_grad = np.zeros((nn_params.shape[0]))

step = np.zeros((nn_params.shape[0]))

e = 1e-4

for i in range(nn_params.shape[0]):

step[i] = e

loss1 = nnCostFunction(nn_params-step.reshape(-1,1), input_layer_size, hidden_layer_size,

num_labels, X, y,

Lambda)

loss2 = nnCostFunction(nn_params+step.reshape(-1,1), input_layer_size, hidden_layer_size,

num_labels, X, y,

Lambda)

num_grad[i] = (loss2-loss1)/(2*e)

step[i]=0

# 显示两列比较

res = np.hstack((num_grad.reshape(-1,1),grad.reshape(-1,1)))

print res

权重的随机初始化

神经网络不能像逻辑回归那样初始化theta为0,因为若是每条边的权重都为0,每个神经元都是相同的输出,在反向传播中也会得到同样的梯度,最终只会预测一种结果。

所以应该初始化为接近0的数

实现代码

# 随机初始化权重theta

def randInitializeWeights(L_in,L_out):

W = np.zeros((L_out,1+L_in)) # 对应theta的权重

epsilon_init = (6.0/(L_out+L_in))**0.5

W = np.random.rand(L_out,1+L_in)*2*epsilon_init-epsilon_init # np.random.rand(L_out,1+L_in)产生L_out*(1+L_in)大小的随机矩阵

return W

预测

正向传播预测结果

实现代码

# 预测

def predict(Theta1,Theta2,X):

m = X.shape[0]

num_labels = Theta2.shape[0]

#p = np.zeros((m,1))

'''正向传播,预测结果'''

X = np.hstack((np.ones((m,1)),X))

h1 = sigmoid(np.dot(X,np.transpose(Theta1)))

h1 = np.hstack((np.ones((m,1)),h1))

h2 = sigmoid(np.dot(h1,np.transpose(Theta2)))

'''

返回h中每一行最大值所在的列号

- np.max(h, axis=1)返回h中每一行的最大值(是某个数字的最大概率)

- 最后where找到的最大概率所在的列号(列号即是对应的数字)

'''

#np.savetxt("h2.csv",h2,delimiter=',')

p = np.array(np.where(h2[0,:] == np.max(h2, axis=1)[0]))

for i in np.arange(1, m):

t = np.array(np.where(h2[i,:] == np.max(h2, axis=1)[i]))

p = np.vstack((p,t))

return p

输出结果

梯度检查:

python bp神经网络代码实现预测,用Python实现BP神经网络(附代码)_第4张图片

随机显示100个手写数字

python bp神经网络代码实现预测,用Python实现BP神经网络(附代码)_第5张图片

显示theta1权重

python bp神经网络代码实现预测,用Python实现BP神经网络(附代码)_第6张图片

训练集预测准确度

python bp神经网络代码实现预测,用Python实现BP神经网络(附代码)_第7张图片

归一化后训练集预测准确度

b3dba5d16b927e3bc753591f0dc89431.png

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

你可能感兴趣的:(python,bp神经网络代码实现预测)