深度学习中权重初始化的重要性

深度学习模型中的权重初始化经常被人忽略,而事实上这是非常重要的一个步骤,模型的初始化权重的好坏关系到模型的训练成功与否,以及训练速度是否快速,效果是否更好等等,这次我们专门来看看深度学习中的权重初始化问题。本文参考斋藤康毅的书籍《深度学习入门》。

首先,一般的权重初始化是用高斯分布生成的值再乘以0.01后得到的,也就是均值为0,标准差为0.01的一组随机数。模型会使用初始化之后的权重再进行训练。

np.random.randn(10, 100)*0.01

为什么权重值不能设置为全零,或者完全一样的值呢?因为在误差反向传播中,所有的权重值都会进行相同的更新,假设一个两层的神将网络,如果第一层和第二层的权重为0,那么第二层的神经元中全部输入相同的值,这意味着反向传播时,第二层的权重全部都会进行相同的更新,这样的话,权重都会更新为相同的值,也就是所有分量的权重都一样,因此损失将不会下降,也就无法进行学习了。

下面我们来看看进行不同的权重初始化后,网络激活值的变化。

假设有一个五层的神经网络,激活函数使用sigmoid函数,用直方图画出每层的激活值的数据分布:

import matplotlib.pyplot as pltdef sigmoid(x):return 1/(1+np.exp(-x))x = np.random.randn(1000, 100) # 1000个数据,每个数据100维
node_num = 100 # 隐藏层的节点数
hidden_layer_size = 5 # 隐藏层的数量
activations = {} #激活值的结果for i in range(hidden_layer_size):# i=0的时候先计算激活值,往后每次的输入值x都是上一次激活值结果if i!=0:x = activations[i-1]w = np.random.randn(node_num, node_num)*1 # 权重初始化z = np.dot(x,w)a = sigmoid(z) # sigmoid函数activations[i] = a
# 绘制直方图
for i, a in activations.items():plt.subplot(1, len(activations), i+1)plt.title(str(i+1)+"-layer")plt.hist(a.flatten(), 30, range=(0,1))
plt.show()

目前我们的初始化权重是高斯分布的随机数,也就是均值为0,标准差为1,得到的结果如下图所示:

可以看到随着层数加深,越来越的多激活值集中到0和1,从而导致它们的导数接近0,因此偏向0和1的数据分布会造成反向传播中梯度值不断减小,从而产生“梯度消失”问题,特别是当神经网络的层数不断加深时,这种问题会更明显。

下面我们把初始化权重变成均值为0,标准差为0.01的高斯分布,得到的结果如下:

w = np.random.randn(node_num, node_num)*0.01 # 权重初始化

这个分布的问题是,随着神经网络层数的加深,越来越多的激活值趋于相同,如果这样下去的话,所有神经元输出的值都一样,那就是说再多的神经元效果都和一个神经元一样,因此会出现“表现力受限”问题,效果一样不好。比较理想的情况是,各层的激活值的分布都要求有适当的广度。为什么呢?因为通过在各层间传递多样性的数据,神经网络可以进行高效的学习。反过来,如果传递的是有所偏向的数据,就会出现梯度消失或者“表现力受限”的问题,导致学习可能无法顺利进行。

对于sigmoid激活值,我们采用Xavier初始值是比较理想的权重初始化方式。Xavier初始值就是,假设前一层的节点数是n,则初始权重使用均值为0,标准差为1/开根号(n)的高斯分布。

w = np.random.randn(node_num, node_num)*1/np.sqrt(n) # xavier权重初始化

可以看到,此时输出的激活值分布具有比较高的广度,所以sigmoid的表现力不受限制,可以进行高效的学习。

不过这种Xavier权重初始化比较适合以sigmoid为激活函数的神经网络。实际应用中,我们更多的是使用ReLU作为神经网络的激活函数,当使用ReLU函数的初始化权重比较适合采用何凯明提出的kaiming权重初始化方法,kaiming初始化其实就是在xavier的基础上,根号内部乘以了2。因为ReLU函数在小于0的时候,值为0,为了增加它的广度,所以乘以了2倍系数。

w = np.random.randn(node_num, node_num)*np.sqrt(2/node_num) # kaiming权重初始化

下面我们来看看这两种权重初始化的不同效果。

def relu(x):return np.maximum(x,0)x = np.random.randn(1000, 100) # 1000个数据,每个数据100维
node_num = 100 # 隐藏层的节点数
hidden_layer_size = 5 # 隐藏层的数量
activations = {} # 激活值的结果for i in range(hidden_layer_size):# i=0的时候先计算激活值,往后每次的输入值x都是上一次激活值结果if i!=0:x = activations[i-1]#w = np.random.randn(node_num, node_num)*0.01 # 一般权重初始化w = np.random.randn(node_num, node_num)/np.sqrt(node_num) # xavier权重初始化#w = np.random.randn(node_num, node_num)*np.sqrt(2/node_num) # kaiming权重初始化z = np.dot(x,w)a = relu(z) # 激活函数activations[i] = a
# 绘制直方图
for i, a in activations.items():plt.subplot(1, len(activations), i+1)plt.title(str(i+1)+"-layer")plt.hist(a.flatten(), 30, range=(0,1))
plt.show()

随着层数的假设,xavier权重初始化,会使得激活值越来越靠近0,从而出现梯度消失的问题。

w = np.random.randn(node_num, node_num)*np.sqrt(2/node_num) # kaiming权重初始化

当我们采用kaiming初始化的时候,结果如下: 

可以看到,不论层数多深,激活值总是均匀分布在0~1之间,因此适合通过反向传播进行学习。

下面我们比较一下采用(0,0.01)的高斯分布做权重初始化和用xavier做权重初始化在多层感知机上的训练效果。假设我有一个线性不可分数据集:

import numpy as np
import matplotlib.pyplot as plt# 构建非线性可分数据集
def create_dataset():np.random.seed(1)m = 400 # 数据量N = int(m/2) # 每一类数据的个数dim = 2 # 数据维度X = np.zeros((m,dim))Y = np.zeros((m,1), dtype='uint8')a = 4# 生成数据for j in range(2):ix = range(N*j,N*(j+1))t = np.linspace(j*3.12,(j+1)*3.12,N)+np.random.randn(N)*0.2r = a*np.sin(4*t) + np.random.randn(N)*0.2X[ix] = np.c_[r*np.sin(t), r*np.cos(t)]Y[ix] = jX = X.TY = Y.Treturn X,YX,Y = create_dataset()
X = X.T
Y = Y.T
print(Y.shape)
print(X.shape)
plt.scatter(X[:, 0], X[:, 1], c=Y, cmap=plt.cm.Paired) # 画出数据

生成的图像如下,我们需要对该数据集进行分类,无法用单一的线性函数去分类,因此可以用神经网络(多层感知机)去尝试分类。

下面我们用numpy编写一个多层感知机:

def net(X,Y):# 定义网络结构,X数据,Y标签n = X.shape[0]num_hidden = 4 # 隐藏层的神经元个数m = Y.shape[0]return (n, num_hidden, m)def initialize_parameters(n, num_hidden, m):# 初始化参数w1 = np.random.randn(num_hidden, n)*0.01 # 从输入到隐藏层权重,随机初始化#w1 = np.zeros((num_hidden, n))#w1 = np.random.randn(num_hidden, n)/np.sqrt(n) # xavier权重初始化b1 = np.zeros((num_hidden, 1)) # 从输入到隐藏层偏置,初始化为零w2 = np.random.randn(m, num_hidden)*0.01 # 从隐藏层到输出层权重,随机初始化#w2 = np.zeros((m, num_hidden))#w2 = np.random.randn(m, num_hidden)/np.sqrt(num_hidden) # xavier权重初始化b2 = np.zeros((m, 1)) # 从隐藏层到输出层权重,初始化为零parameters = {'w1':w1,'w2':w2,'b1':b1,'b2':b2}return parametersdef sigmoid(x): # sigmoid激活函数s = 1 / (1 + np.exp(-x))return sdef forward(X,parameters): # 前向运算w1 = parameters['w1']b1 = parameters['b1']w2 = parameters['w2']b2 = parameters['b2']z1 = np.dot(w1,X)+b1#print("z1.shape:",z1.shape)a1 = np.tanh(z1)z2 = np.dot(w2,a1)+b2#print("z2.shape:",z2.shape)a2 = sigmoid(z2)cache = {'z1':z1,'a1':a1,'z2':z2,'a2':a2}return cachedef backward_propagation(parameters, cache, X, Y): # 反向传播m = X.shape[1] w1 = parameters['w1']w2 = parameters['w2']b1 = parameters['b1']b2 = parameters['b2']a1 = cache['a1']a2 = cache['a2']# 反向传播,根据吴恩达教程的公式推导dz2 = a2-Ydw2 = 1/m*np.dot(dz2,a1.T)db2 = 1/m*np.sum(dz2, axis=1, keepdims=True)dz1 = np.dot(w2.T,dz2)*(a1-np.power(a1,2))dw1 = 1/m*np.dot(dz1,X.T)db1 = 1/m*np.sum(dz1, axis=1, keepdims=True)grads = {'dw1':dw1,'db1':db1,'dw2':dw2,'db2':db2}return gradsdef loss(z2,Y): # 损失值计算m = Y.shape[1] # 列向量的数量loss = np.log(z2)*Y + np.log(1-z2)*(1-Y)loss = -1/m * np.sum(loss)loss = np.squeeze(loss)return lossdef update_weights(parameters, grads, lr): # 权重更新w1 = parameters['w1']w2 = parameters['w2']b1 = parameters['b1']b2 = parameters['b2']dw1 = grads['dw1']dw2 = grads['dw2']db1 = grads['db1']db2 = grads['db2']# 参数更新w1 -= lr*dw1b1 -= lr*db1w2 -= lr*dw2b2 -= lr*db2parameters = {'w1':w1,'w2':w2,'b1':b1,'b2':b2}return parametersdef train(X, Y):x,n_h,y = net(X,Y) # 构建网络并获取到输入和输出节点数print("x:{},n_h:{},y:{}".format(x,n_h,y))parameters = initialize_parameters(x,n_h,y)w1 = parameters['w1']b1 = parameters['b1']w2 = parameters['w2']b2 = parameters['b2']for i in range(100000):cache = forward(X, parameters)a2 = cache['a2']cost = loss(a2,Y)grads = backward_propagation(parameters, cache, X, Y)parameters = update_weights(parameters, grads, lr=0.001)if i%10000==0:print("================",cost)return parametersX,Y = create_dataset()
parameters = train(X, Y)
print("w1:{},b1:{},w2:{},b2:{}".format(parameters['w1'],parameters['b1'],parameters['w2'],parameters['b2']))

当我们使用(0,0.01)的高斯分布做权重初始化后,训练损失值如下:

================ 0.6931125167719424
================ 0.6929599436150587
================ 0.6928419798609118
================ 0.6927533221282552
================ 0.6926665006259728
================ 0.6925442899224475
================ 0.6923418054740101
================ 0.6919987185837877
================ 0.6914315483776486
================ 0.6905487939721282

可以看到损失值几乎不下降,无法训练,如果使用xavier权重初始化后,训练损失值如下:

================ 0.7088135432785352
================ 0.6196813019938332
================ 0.5770935283638372
================ 0.5129738885023197
================ 0.46283119618617713
================ 0.4218237851915789
================ 0.3914602876997484
================ 0.3705175596789437
================ 0.354306018670427
================ 0.3409172900251636

可以看到,使用(0,0.01)的高斯分布做权重初始化训练很难进行下去,而使用xavier做权重初始化后,训练会非常快速,损失值平稳的下降。最终分类结果也比较不错:

其实,我们平时工作中倒是不用特别担心这一点,因为我们现在一般采用pytorch之类的深度学习框架进行代码编写,很少从头开始构建深度学习模型,pytorch之类的框架中会对权重一个默认的初始化,而且效果都不错。不过权重初始化依然是一个非常重要的概念,对于一些特殊的模型,可能需要我们自己手动做权重初始化才能更好的训练。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/660598.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

数据库管理-第179期 分库分表vs分布式(20240430

数据库管理179期 2024-04-30 数据库管理-第179期 分库分表vs分布式(20240430)1 分库分表1.1 分库1.2 分表1.3 组合1.4 问题 2 分布式3 常见分布式数据库4 期望总结 数据库管理-第179期 分库分表vs分布式(20240430) 作者&#xff1…

iBarcoder for Mac:一站式条形码生成软件

在数字化时代,条形码的应用越来越广泛。iBarcoder for Mac作为一款专业的条形码生成软件,为用户提供了一站式的解决方案。无论是零售、出版还是物流等行业,iBarcoder都能轻松应对,助力用户实现高效管理。 iBarcoder for Mac v3.14…

【Hadoop】-Hive客户端:HiveServer2 Beeline 与DataGrip DBeaver[14]

HiveServer2 & Beeline 一、HiveServer2服务 在启动Hive的时候,除了必备的Metastore服务外,我们前面提过有2种方式使用Hive: 方式1: bin/hive 即Hive的Shell客户端,可以直接写SQL方式2: bin/hive --…

使 Elasticsearch 和 Lucene 成为最佳向量数据库:速度提高 8 倍,效率提高 32 倍

作者:来自 Elastic Mayya Sharipova, Benjamin Trent, Jim Ferenczi Elasticsearch 和 Lucene 成绩单:值得注意的速度和效率投资 我们 Elastic 的使命是将 Apache Lucene 打造成最佳的向量数据库,并继续提升 Elasticsearch 作为搜索和 RAG&a…

SQLite如何处理CSV 虚拟表(三十七)

返回:SQLite—系列文章目录 上一篇:SQLite的DBSTAT 虚拟表(三十六) 下一篇:SQLite的扩展函数Carray()表值函数(三十八) ​ RFC4180格式是一种文本文件格式,被用于表格数据间的交互,也可将表格数据转化…

人工智能分割分类model:nnUnet-paddle

文章目录 神经网络nnUnet和paddle都需要在Ubuntu下进行安装PaddleProject 神经网络 开源来自https://github.com/MIC-DKFZ/nnUNet 自建了仓库,但还不会用 来自 mmsegmentation有空去了解 . MICCAI 2020 也是用到这个网络 paddle上的是不是不能用… nnUnet和pad…

Qt5画饼图、圆环图、极地图、折线图

(1)Qt5Chart应用 (2)展现形式 (3)下载地址: https://download.csdn.net/download/hgaohr1021/89247166 Qt5画饼图、圆环图、极地图、折线图

hive-row_number() 和 rank() 和 dense_rank()

row_number() 是无脑排序 rank() 是相同的值排名相同,相同值之后的排名会继续加,是我们正常认知的排名,比如学生成绩。 dense_rank()也是相同的值排名相同,接下来的排名不会加。不会占据排名的坑位。

centos7安装真的Redmine-5.1.2+ruby-3.0.0

下载redmine-5.1.2.tar.gz,上传到/usr/local/目录下 cd /usr/local/ tar -zxf redmine-5.1.2.tar.gz cd redmine-5.1.2 cp config/database.yml.example config/database.yml 配置数据连接 #编辑配置文件 vi config/database.yml #修改后的内容如下 product…

智能体可靠性的革命性提升,揭秘知识工程领域的参考架构新篇章

引言:知识工程的演变与重要性 知识工程(Knowledge Engineering,KE)是一个涉及激发、捕获、概念化和形式化知识以用于信息系统的过程。自计算机科学和人工智能(AI)历史以来,知识工程的工作流程因…

【百度Apollo】探索自动驾驶:百度Apollo视觉感知模块的实践与创新

🎬 鸽芷咕:个人主页 🔥 个人专栏: 《linux深造日志》《粉丝福利》 ⛺️生活的理想,就是为了理想的生活! 文章目录 引入一、百度Apollo视觉感知模块概述二、启动感知模块步骤一:进入 Docker 环境并启动 Dreamview步骤二…

vue 前端读取Excel文件并解析

前端读取Excel文件并解析 前端如何解释Excel呢 平时项目中对于Excel的导入解析是很常见的功能,一般都是放在后端执行;但是也有特殊的情况,偶尔也有要求说前端执行解析,判空,校验等,最后组装成后端接口想要的…