前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >一文搞懂深度网络初始化(Xavier and Kaiming initialization)

一文搞懂深度网络初始化(Xavier and Kaiming initialization)

作者头像
狼啸风云
修改2022-09-03 19:03:59
9.3K0
修改2022-09-03 19:03:59
举报
文章被收录于专栏:计算机视觉理论及其实现

Xavier Initialization

早期的参数初始化方法普遍是将数据和参数normalize为高斯分布(均值0方差1),但随着神经网络深度的增加,这方法并不能解决梯度消失问题。

Xavier初始化的作者,Xavier Glorot,在Understanding the difficulty of training deep feedforward neural networks论文中提出一个洞见:激活值的方差是逐层递减的,这导致反向传播中的梯度也逐层递减。要解决梯度消失,就要避免激活值方差的衰减,最理想的情况是,每层的输出值(激活值)保持高斯分布。

We initialized the biases to be 0 and the weights W_{i j} at each layer with the following commonly used heuristic:

W_{i j} \sim U\left[-\frac{1}{\sqrt{n}}, \frac{1}{\sqrt{n}}\right]

因此,他提出了Xavier初始化:bias初始化为0,为Normalize后的参数乘以一个rescale系数:1 / \sqrt{n} ,n是输入参数的个数。

公式的推导过程大致如下:

y=a x+b=W \vec{x}+\vec{b}=\vec{w}_{1} x_{1}+\vec{w}_{2} x_{2}+\ldots+\vec{w}_{n} x_{n}+\vec{b}

\operatorname{var}(y)=\operatorname{var}\left(\vec{w}_{1} x_{1}+\ldots+\vec{w}_{n} x_{n}+\vec{b}\right)=\operatorname{var}\left(\vec{w}_{1} x_{1}\right)+\ldots+\operatorname{var}\left(\vec{w}_{n} x_{n}\right)

\operatorname{var}\left(\vec{w}_{i} x_{i}\right)=E\left(x_{i}\right)^{2} \operatorname{var}\left(\vec{w}_{i}\right)+E\left(\vec{w}_{i}\right)^{2} \operatorname{var}\left(x_{i}\right)+\operatorname{var}\left(\vec{w}_{i}\right) \operatorname{var}\left(x_{i}\right)

因为E(期望)等于均值,而输入数据(x)和参数(W)的均值都是0,因此,

\operatorname{var}\left(\vec{w}_{i} x_{i}\right)=\operatorname{var}\left(\vec{w}_{i}\right) \operatorname{var}\left(x_{i}\right)

\operatorname{var}(y)=\operatorname{var}\left(\vec{w}_{1}\right) \operatorname{var}\left(x_{1}\right)+\operatorname{var}\left(\vec{w}_{2}\right) \operatorname{var}\left(x_{2}\right)+\ldots+\operatorname{var}\left(\vec{w}_{n}\right) \operatorname{var}\left(x_{n}\right)

又因为xW恒等分布(方差都是1),因此\operatorname{var}(y)=N * \operatorname{var}\left(\vec{w}_{i}\right) \operatorname{var}\left(x_{i}\right)

我们的目标是\operatorname{var}(y)=\operatorname{var}(x) ,因此\operatorname{var}(y)=N * \operatorname{var}\left(\vec{w}_{i}\right) \operatorname{var}\left(x_{i}\right)s t d=\sqrt{v a r}, s t d\left(\vec{w}_{i}\right)=1 / \sqrt{N}

代码语言:javascript
复制
def linear(x, w, b): return x @ w + b

def relu(x): return x.clamp_min(0.)

nh = 50
W1 = torch.randn(784, nh)
b1 = torch.zeros(nh)
W2 = torch.randn(nh, 1)
b2 = torch.zeros(1)

z1 = linear(x_train, W1, b1)
print(z1.mean(), z1.std())

tensor(-0.8809) tensor(26.9281)

这是个简单的线性回归模型:y=a x+b(W_1, b_1)(W_2, b_2)分别是隐层和输出层的参数,W_1/W_2初始化为高斯分布,b_1/b_2初始为0。果然,第一个linear层的输出值(z_1)的均值和标准差就已经发生了很大的变化。如果后续使用sigmoid作为激活函数,那梯度消失就会很明显。

现在我们按照Xavier的方法来初始化参数:

代码语言:javascript
复制
W1 = torch.randn(784, nh) * math.sqrt(1 / 784)
b1 = torch.zeros(nh)
W2 = torch.randn(nh, 1) * math.sqrt(1 / nh)
b2 = torch.zeros(1)

z1 = linear(x_train, W1, b1)
print(z1.mean(), z1.std())

tensor(0.1031) tensor(0.9458)

a1 = relu(z1)
a1.mean(), a1.std()

(tensor(0.4272), tensor(0.5915))

参数经过Xavier初始化后,linear层的输出值的分布没有大的变化(U[0.1031,0.9458] ),依旧接近高斯分布,但是好景不长,relu的激活值分布就开始跑偏了(U[0.4272,0.5915] )。

Kaiming Initialization

Xavier初始化的问题在于,它只适用于线性激活函数,但实际上,对于深层神经网络来说,线性激活函数是没有价值,神经网络需要非线性激活函数来构建复杂的非线性系统。今天的神经网络普遍使用relu激活函数。

Kaiming初始化的发明人kaiming he,在Delving Deep into Rectifiers: Surpassing Human-Level Performance on ImageNet Classification论文中提出了针对relu的kaiming初始化。

因为relu会抛弃掉小于0的值,对于一个均值为0的data来说,这就相当于砍掉了一半的值,这样一来,均值就会变大,前面Xavier初始化公式中E(x)=mean=0的情况就不成立了。根据新公式的推导,最终得到新的rescale系数:\sqrt{2 / n} 。更多细节请看论文的section 2.2

代码语言:javascript
复制
W1 = torch.randn(784, nh) * math.sqrt(2 / 784)
b1 = torch.zeros(nh)
W2 = torch.randn(nh, 1) * math.sqrt(2 / nh)
b2 = torch.zeros(1)

z1 = linear(x_train, W1, b1)
a1 = relu(z1)
a1.mean(), a1.std()

(tensor(0.4553), tensor(0.7339))

可以看到,Kaiming初始化的表现要优于Xavier初始化,relu之后的输出值标准差还有0.7339(浮动可以达到0.8+)。

实际上,Kaiming初始化已经被Pytorch用作默认的参数初始化函数。

代码语言:javascript
复制
import torch.nn.init as init

W1 = torch.zeros(784, nh)
b1 = torch.zeros(nh)
W2 = torch.zeros(nh, 1)
b2 = torch.zeros(1)

init.kaiming_normal_(W1, mode='fan_out', nonlinearity='relu')
init.kaiming_normal_(W2, mode='fan_out')
z1 = linear(x_train, W1, b1)
a1 = relu(z1)
print("layer1: ", a1.mean(), a1.std())
z2 = linear(a1, W2, b2)

layer1:  tensor(0.5583) tensor(0.8157)
tensor(1.1784) tensor(1.3209)

现在,方差的问题已经解决了,接下来就是均值不为0的问题。因为在x轴上平移data并不会影响data的方差,因此,如果把relu的激活值左移5,结果会如何?

代码语言:javascript
复制
def linear(x, w, b):
  return x @ w + b

def relu(x):
  return x.clamp_min(0.) - 0.5

def model(x):
  x = relu(linear(x, W1, b1))
  print("layer1: ", x.mean(), x.std())
  x = relu(linear(x, W2, b2))
  print("layer2: ", x.mean(), x.std())
  x = linear(x, W3, b3)
  print("layer3: ", x.mean(), x.std())
  return x

nh = [100, 50]
W1 = torch.zeros(784, nh[0])
b1 = torch.zeros(nh[0])
W2 = torch.zeros(nh[0], nh[1])
b2 = torch.zeros(nh[1])
W3 = torch.zeros(nh[1], 1)
b3 = torch.zeros(1)

init.kaiming_normal_(W1, mode='fan_out')
init.kaiming_normal_(W2, mode='fan_out')
init.kaiming_normal_(W3, mode='fan_out')
_ = model(x_train)

layer1:  tensor(0.0383) tensor(0.7993)
layer2:  tensor(0.0075) tensor(0.7048)
layer3:  tensor(-0.2149) tensor(0.4493)

结果出乎意料的好,这个三层的模型在没有添加batchnorm的情况下,每层的输入值和输出值都接近高斯分布,虽然数据方差是会逐层递减,但相比normalize初始化和Xavier初始化要好很多。

最后,因为Kaiming初始化是pytorch的默认初始化函数,因此我又用pytorch提供的nn.Linear()和nn.Relu()来构建相同的模型对比测试,结果是大跌眼镜。

代码语言:javascript
复制
class Model(nn.Module):
  def __init__(self):
    super().__init__()
    self.lin1 = nn.Linear(784, nh[0])
    self.lin2 = nn.Linear(nh[0], nh[1])
    self.lin3 = nn.Linear(nh[1], 1)
    self.relu = nn.ReLU()
  
  def forward(self, x):
    x = self.relu(self.lin1(x))
    print("layer 1: ", x.mean().item(), x.std().item())
    x = self.relu(self.lin2(x))
    print("layer 2: ", x.mean().item(), x.std().item())
    x = self.relu(self.lin3(x))
    print("layer 3: ", x.mean().item(), x.std().item())
    return x

m = Model()
_ = m(x_train)

layer 1:  0.2270725518465042 0.32707411050796
layer 2:  0.033514849841594696 0.23475737869739532
layer 3:  0.013271240517497063 0.09185370802879333

可以看到,第三层的输出已经均值为0、方差为0。去看nn.Linear()类的代码时会看到,它在做初始化时会传入参数a=math.sqrt(5)。我们知道,当输入为负数时,leaky relu的梯度为[0, \infty]x=\lambda x ,参数a就是这个。虽然kaiming_uniform_()的默认网络要使用的激活函数是leaky relu,但a默认值为0,此时leaky relu就等于relu。但现在数据存在负数,因此,mean相比relu模型更接近于0,甚至E(x) > 0的假设都不成立了,因此,rescale系数就不准确了,nn.Linear()才会有这样的表现。

代码语言:javascript
复制
def reset_parameters(self):
    init.kaiming_uniform_(self.weight, a=math.sqrt(5))

END

本文通过Xavier和Kaiming初始化来展现了参数初始化的重要性,因为糟糕的初始化容易让神经网络陷入梯度消失的陷阱中。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2020/02/15 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Xavier Initialization
  • Kaiming Initialization
  • END
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档