Datawhale X 李宏毅苹果书 AI夏令营 《深度学习详解》第三章 深度学习基础

3.1 局部极小值与鞍点

        1、临界点及其种类

         在我们的训练中是会存在梯度下降失效的问题的

        提到梯度为零的时候,大家最先想到的可能就是局部极小值(local minimum) ,但其实损失不是只在局部极小值的梯度是零,还有其他可能会让梯度是零的点,比如鞍点(saddle point)。鞍点其实就是梯度是零且区别于局部极小值和局部极大值(localmaximum)的点,我们将梯度为零的点统称为临界点(critical point)

        2、判断临界点种类的方法

        判断一个临界点到底是局部极小值还是鞍点需要知道损失函数的形状。虽然无法完整知道整个损失函数的样子,但是如果给定某一组参数,比如\theta{}',在\theta {}'附近的损失函数是有办法写出来的:L\left ( \theta \right )\approx L\left ( \theta {}' \right )+\left ( \theta -\theta {}' \right )^{T}g+\frac{1}{2}\left ( \theta -\theta {}' \right )^{T}H\left ( \theta -\theta {}' \right ),我们利用泰勒级数近似(Tayler series appoximation)得到这个近似公式。

  • 第一项L\left ({\theta }' \right )告诉我们,当\theta\theta {}'很近的时候,L\left ( \theta \right )应该跟L\left ( \theta {}' \right )比较接近
  • 第二项\left ( \theta -{\theta }' \right )^{T}g中,g代表梯度,它是一个向量,可以弥补L\left ( {\theta }' \right )L\left ( \theta \right )之间的差距。有时候梯度g会写成\bigtriangledown L\left ( {\theta }' \right )g_{i}是向量g的第 i 个元素,就是L关于\theta的第 i 个元素的微分,即g_{i}=\frac{\partial L\left ( {\theta }' \right )}{\partial \theta _{i}}
  • 第三项\frac{1}{2}\left ( \theta -{\theta }' \right )^{T}H\left ( \theta -{\theta }' \right )海森矩阵(Hessian matrix)H有关。H里面放的是L的二次微分,它第 i 行,第 j 列的值H_{ij}就是把\theta的第 i 个元素对L\left ( {\theta }' \right )作微分,再把\theta的第 j 个元素对\frac{\partial L\left ( {\theta }' \right )}{\partial \theta _{i}}作微分后的结果,即H_{ij}=\frac{\partial ^{2}}{\partial \theta _{i}\partial \theta _{j}}L\left ( {\theta }' \right )

        在临界点,梯度g为零,因此\left ( \theta -{\theta }' \right )^{T}g为零。所以在临界点的附近,损失函数可被近似为L\left ( \theta \right )\approx L\left ( \theta \right )+\frac{1}{2}\left ( \theta -{\theta }' \right )H\left ( \theta -{\theta }' \right ),我们可以根据这个式子的第二项来判断在{\theta }'附近的误差表面(error surface)到底是什么样子的。为了符号简洁,我们用向量v来表示\left ( \theta -{\theta }' \right )\left ( \theta -{\theta }' \right )^{T}H\left ( \theta -{\theta }' \right )可改写为v^{T}Hv,有如下三种情况:

  • 对所有vv^{T}Hv> 0。这意味着对任意\thetaL\left ( \theta \right )> L\left ( {\theta }' \right )。这代表 L(θ′) 是附近的一个最低点,所以它是局部极小值
  • 对所有vv^{T}Hv< 0。这意味着对任意\thetaL\left ( \theta \right )< L\left ( {\theta }' \right )。这代表 L(θ′) 是附近的一个最高点,所以它是局部极大值
  • 对于vv^{T}Hv有时候大于零,有时候小于零。这意味着在{\theta }'附近,有时候L\left ( \theta \right )< L\left ( {\theta }' \right ),有时候L\left ( \theta \right )> L\left ( {\theta }' \right ),所以它既不是局部极大值,也不是局部极小值,而是鞍点

        3、逃离鞍点的方法

        最小值比例(minimum ratio)定义为: 最小值比例=正特征值数量/总特征值数量。从经验上来看,遇到局部极小值的可能性极小,多数的时候,我们训练到一个梯度很小的地方,参数不再更新,往往只是遇到了鞍点。

3.2 批量和动量

        事实上,在把数据分为批量的时候,我们还会进行随机打乱(shuffle)。随机打乱有很多不同的做法,一个常见的做法是在每一个回合开始之前重新划分批量,也就是说,每个回合的批量的数据都不一样。 

        1、批量大小对梯度下降法的影响

         假设我们现在有20笔数据,下面出现两种情况:

  • 批量大小为训练数据的大小。这种使用全批量(fullbatch)的数据来更新参数的方法即批量梯度下降法(Batch Gradient Descent,BGD),此时模型必须把 20 笔训练数据都看完,才能够计算损失和梯度,参数才能够更新一次。

  • 批量大小等于 1。此时使用的方法即随机梯度下降法(Stochastic Gradient Descent,SGD),也称为增量梯度下降法。批量大小等于 1 意味着只要取出一笔数据即可计算损失、更新一次参数。如果总共有 20 笔数据,那么在每一个回合里面,参数会更新 20 次。用一笔数据算出来的损失相对带有更多噪声,因此其更新的方向是曲折的

         实际上,考虑并行运算,批量梯度下降花费的时间不一定更长;对于比较大的批量,计算损失和梯度花费的时间不一定比使用小批量的计算时间长。

         大的批量更新比较稳定,小的批量的梯度的方向是比较有噪声的(noisy)。但实际上有噪声的的梯度反而可以帮助训练,如果拿不同的批量来训练模型来做图像识别问题,得到的结果是:批量大小越大,验证集准确率越差。这个是优化的问题,大的批量大小优化可能会有问题,小的批量大小优化的结果反而是比较好的。

        2、动量法

         动量法(momentum method)是另外一个可以对抗鞍点或局部最小值的方法。

        一般的梯度下降(vanilla gradient descent)初始参数为\theta _{0},计算一下梯度,计算完梯度后,往梯度的反方向去更新参数\theta _{1}=\theta _{0}-\eta g_{0}。有了新的参数\theta _{1}后,再计算一次梯度,再往梯度的反方向更新一次参数,以此类推。

        引入动量后,每次在移动参数的时候,不是只往梯度的反方向来移动参数,而是根据梯度的反方向加上前一步移动的方向决定移动方向。过程如下: 

3.3 自适应学习率

        训练一个网络,训练到后来发现损失不再下降的时候,有时候不是卡在局部最小值或鞍点,只是单纯的损失无法再下降。最原始的梯度下降连简单的误差表面都做不好,因此需要更好的梯度下降的版本。在梯度下降里面,所有的参数都是设同样的学习率,这显然是不够的,应该要为每一个参数定制化学习率,即引入自适应学习率(adaptive learning rate)的方法,给每一个参数不同的学习率。如果在某一个方向上,梯度的值很小,非常平坦,我们会希望学习率调大一点;如果在某一个方向上非常陡峭,坡度很大,我们会希望学习率可以设得小一点。

        1、AdaGrad

        AdaGrad(Adaptive Gradient)是典型的自适应学习率方法,其能够根据梯度大小自动调整学习率。AdaGrad 可以做到梯度比较大的时候,学习率就减小,梯度比较小的时候,学习率就放大。 

        梯度下降更新某个参数\theta _{t}^{i}的过程为:\theta _{t+1}^{i}\leftarrow \theta _{t}^{i}-\eta g_{t}^{i}

        \theta _{t}^{i}在第 t 个迭代的值减掉在第 t 个迭代参数 i 算出来的梯度:g_{t}^{i}=\frac{\partial L}{\partial \theta ^{i}}|_{\theta =\theta _{t}}g_{t}^{i}代表在第 t 个迭代,即\theta = \theta _{t}时, 参数\theta ^{i}对损失L的微分,学习率是固定的。

        现在要有一个随着参数定制化的学习率,即把原来学习率\eta变成\frac{\eta }{\sigma _{t}^{i}}\theta _{t+1}^{i}\leftarrow \theta _{t}^{i}-\frac{\eta }{\sigma _{t}^{i}}g_{t}^{i},其中\sigma _{t}^{i}的上标为 i,这代表参数\sigma与 i 相关,不同的参数的\sigma不同。\sigma _{t}^{i}的下标为 t,这代表参数\sigma与迭代相关,不同的迭代也会有不同的\sigma。学习率从\eta改成\frac{\eta }{\sigma _{t}^{i}}的时候,学习率就变得参数相关(parameter dependent)

         参数相关的一个常见的类型是算梯度的均方根(root mean square)

  • 第一次更新参数过程为:\theta _{1}^{i}\leftarrow \theta _{0}^{i}-\frac{\eta }{\sigma _{0}^{i}}g_{0}^{i},其中\theta _{0}^{i}是初始化参数。而\sigma _{0}^{i}的计算过程为:\sigma _{0}^{i}=\sqrt{\left ( g_{0}^{i} \right )^{2}}=\left | g_{0}^{i} \right |,其中g_{0}^{i}是梯度。将\sigma _{0}^{i}的值代入更新的公式可知\frac{g_{0}^{i}}{\sigma _{0}^{i}}的值是 +1 或 −1
  • 第二次更新参数过程为:\theta _{2}^{i}\leftarrow \theta _{1}^{i}-\frac{\eta }{\sigma _{1}^{i}}g_{1}^{i},其中\sigma _{1}^{i}=\sqrt{\frac{1}{2}\left [ \left ( g_{0}^{i} \right )^{2}+\left ( g_{1}^{i} \right )^{2} \right ]}是过去所有计算出来的梯度的平方的平均再开根号,即均方根
  • 以此类推,第 t + 1 次更新参数的时候,有:\theta _{t+1}^{i}\leftarrow \theta _{t}^{i}-\frac{\eta }{\sigma _{t}^{i}}g_{t}^{i}\sigma _{t}^{i}=\sqrt{\frac{1}{t+1}\sum_{i=0}^{t}\left ( g_{t}^{i} \right )^{2}}

        因此有了\sigma _{t}^{i}这一项以后,就可以随着梯度的不同,每一个参数的梯度的不同,来自动调整学习率的大小。

        2、RMSProp

         同一个参数需要的学习率,也会随着时间而改变,于是就有了一个新的方法———RMSprop(Root Mean Squared propagation)

  • 第一次更新过程为:\theta _{1}^{i}\leftarrow \theta _{0}^{i}-\frac{\eta }{\sigma _{0}^{i}}g_{0}^{i}\sigma _{0}^{i}=\sqrt{\left ( g_{0}^{i} \right )^{2}}=\left | g_{0}^{i} \right |
  • 第二次更新过程为:\theta _{2}^{i}\leftarrow \theta _{1}^{i}-\frac{\eta }{\sigma _{1}^{i}}g_{1}^{i}\sigma _{1}^{i}=\sqrt{\alpha \left ( \sigma _{0}^{i} \right )^{2}+\left ( 1-\alpha \right )\left (g _{1}^{i} \right )^{2}}
  • 以此类推,第 t + 1 次更新参数的时候,有:\theta _{t+1}^{i}\leftarrow \theta _{t}^{i}-\frac{\eta }{\sigma _{t}^{i}}g_{t}^{i}\sigma _{t}^{i}=\sqrt{\alpha \left ( \sigma _{t-1}^{i} \right )^{2}+\left ( 1-\alpha \right )\left ( g_{t}^{i} \right ) ^{2}}

        RMSProp 通过\alpha可以决定,g_{t}^{i}相较于之前存在\sigma _{t-1}^{i}里面的g_{1}^{i}, g_{2}^{i}, · · · · · · , g_{t-1}^{i}的重要性有多大。如果使用 RMSprop,就可以动态调整\sigma _{t}^{i}这一项

        3、Adam

         最常用的优化的策略或者优化器(optimizer)Adam(Adaptive moment estimation)。Adam 可以看作 RMSprop 加上动量,其使用动量作为参数更新方向,并且能够自适应调整学习率。PyTorch 里面已经帮我们写好了 Adam 优化器,这个优化器里面有一些超参数需要人为决定,但是往往用 PyTorch 预设的参数就好了,并不需要我们进行改动。

3.4 学习率调度

        之前的学习率调整方法中\eta是一个固定的值,而在学习率调度(learning rate scheduling)\eta跟时间有关。学习率调度中最常见的策略是学习率衰减(learning rate decay),也称为学习率退火(learning rateannealing),随着参数的不断更新,让\eta越来越小

         除了学习率下降以外,还有另外一个经典的学习率调度的方式———预热。预热的方法是让学习率先变大后变小,至于变到多大、变大的速度、变小的速度是超参数。残差网络,BERT 和 Transformer 的训练都使用了预热。

3.5 优化总结

        所以我们从最原始的梯度下降,进化到这一个版本:\theta _{t+1}^{i}\leftarrow \theta _{t}^{i}-\frac{\eta }{\sigma _{t}^{i}}m_{t}^{i},其中m_{t}^{i}是动量

3.6 分类

        1、分类与回归的关系

        回归是输入一个向量x,输出\hat{y},我们希望\hat{y}跟某一个标签y越接近越好,y是要学习的目标。而分类可当作回归来看,输入x后,输出仍然是一个标量\hat{y},它跟正确答案的那个类越接近越好。实际上,在做分类的问题的时候,比较常见的做法是用独热向量表示类,这样类两两之间的距离都是一样的。

        如果y是有多个元素的向量,网络也要产生多个输出,每个输出都是跟目标越接近越好:

        2、带有 softmax 的分类

        按照上述的设定,分类实际过程是:输入x,乘上W,加上b,通过激活函数\sigma,乘上{W}',再加上{b}'得到向量\hat{y}。但实际做分类的时候,往往会把\hat{y}通过 softmax 函数得到{y}',才去计算{y}'\hat{y}之间的距离。

         softmax 的计算:{y_{i}}'=\frac{\exp \left ( y_{i} \right )}{\sum_{j}^{}\exp \left ( y_{i} \right )},其中1> {y_{i}}'> 0\sum_{i}^{}{y_{i}}'=1

        3、分类损失

        当我们把x输入到一个网络里面产生\hat{y}后,通过softmax得到\hat{y},再去计算{y}'y之间的距离e,有两种计算方法,一种是均方误差:e=\sum_{i}^{}\left ( y_{i} -{y_{i}}'\right )^{2},一种是交叉熵:e=-\sum_{i}^{}y_{i}\ln {y_{i}}',当\hat{y}{y}'相同时,可以最小化交叉熵的值,其实就是最大化似然(maximize likelihood),此时均方误差也是最小的。

        从优化的角度来说,相较于均方误差,交叉熵是被更常用在分类上 。

3.7 批量归一化

        如果我们可以给不同的维度,同样的数值范围的话,那我们可能就可以制造比较好的误差表面,让训练变得比较容易一点其实有很多不同的方法,这些不同的方法往往就合起来统称为特征归一化(feature normalization) 

        Z 值归一化(Z-score normalization),也称为标准化(standardization)。假设x^{1}x^{R}是我们所有的训练数据的特征向量,把所有训练数据的特征向量,统统都集合起来。向量x^{1}里面x_{1}^{1}代表x^{1}的第一个元素,x_{1}^{2}代表x^{2}的第一个元素,以此类推。我们把不同笔数据即不同特征向量,同一个维度里面的数值取出来,对于每个维度 i,计算其平均值(mean)m_{i}标准差(standard deviation)\sigma _{i}。接下来就可以进行归一化:\tilde{x}_{i}^{r}\leftarrow \frac{x_{i}^{r}-m_{i}}{\sigma _{i}}

        归一化有个好处,做完归一化以后,这个维度上面的数值就会平均是 0,其方差是 1,所以这一排数值的分布就都会在 0 上下。对每一个维度都做一样的归一化,所有特征不同维度的数值都在 0 上下,可能就可以制造一个比较好的误差表面。所以特征归一化方式往往对训练有帮助,它可以使得在做梯度下降时,损失收敛更快一点,训练更顺利一点。 

        1、考虑深度学习

         \tilde{x}代表归一化的特征,把它丢到深度网络里面,去做接下来的计算和训练。

         如何对z做特征归一化?z可以看成另外一种特征。

        首先计算下z^{1},z^{2}, z^{3}的平均值:\mu =\frac{1}{3}\sum_{i=1}^{3}z^{i},接下来计算标准差:\sigma =\sqrt{\frac{1}{3}\sum_{i=1}^{3}\left ( z^{i}-\mu \right )^{2}},在这个式子里,平方就是指对每一个元素都去做平方,开根号指的是对向量里面的每一个元素开根号。最后,根据计算出的\mu\sigma进行归一化:\tilde{z}^{i}=\frac{z^{i-\mu }}{\sigma },其中,除号代表逐元素的除,即分子分母两个向量对应元素相除。

        接下来可以通过激活函数得到其他向量,但是训练数据非常多,现在一个数据集可能有上百万笔数据,GPU 的显存无法把它整个数据集的数据都加载进去。因此,在实现的时候,我们不会让这一个网络考虑整个训练数据里面的所有样本,而是只会考虑一个批量里面的样本。批量归一化适用于批量大小比较大的时候,批量大小如果比较大,也许这个批量大小里面的数据就足以表示整个数据集的分布。这个时候就不需要对整个数据集做特征归一化,而改成只在一个批量上做特征归一化作为近似。

        在做批量归一化的时候往往还会做如下操作:\hat{z}^{i}=\gamma \bigodot \tilde{z}^{i}+\beta,其中,\bigodot代表逐元素的相乘,\beta\gamma可以想成是网络的参数,需要另外再被学习出来。

        2、测试时的批量归一化

        批量归一化在测试也就是推断(inference)的时候,并不需要做什么特别的处理,PyTorch 已经处理好了。在训练的时候,如果有在做批量归一化,每一个批量计算出来的\mu,\sigma,都会拿出来算移动平均(moving average)。假设现在有各个批量计算出来的\mu ^{1}, \mu ^{2}, \mu ^{3}, · · · · · · , \mu ^{t},则可以计算移动平均:\bar{\mu }\leftarrow p\bar{\mu }+\left ( 1-p \right )\mu ^{t},其中,\bar{\mu }\mu的个平均值,p是因子,这也是一个常数,这也是一个超参数,在 PyTorch 里面,p设 0.1

        3、内部协变量偏移

        原始的批量归一化论文里面提出内部协变量偏移(internal covariate shift)概念

         其实批量归一化不是唯一的归一化方法,还有很多归一化方法,比如批量重归一化(batchrenormalization)层归一化(layer normalization)实例归一化(instance normalization)组归一化(group normalization)权重归一化(weight normalization)谱归一化(spectrum normalization)


hahaha都看到这里了,要是觉得有用的话就辛苦动动小手点个赞吧! 

  • 25
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Pluses

喜欢就给颗糖果吧╰(*°▽°*

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值