神经网络发展历程简单介绍(可逆神经网络详细解析)(1)

来源:PaperWeekly本文约4600字,建议阅读10分钟本文以可逆残差网络作为基础进行分析。

为什么要用可逆网络呢?

  1. 因为编码和解码使用相同的参数,所以 model 是轻量级的。可逆的降噪网络 InvDN 只有 DANet 网络参数量的 4.2%,但是 InvDN 的降噪性能更好。
  2. 由于可逆网络是信息无损的,所以它能保留输入数据的细节信息。
  3. 无论网络的深度如何,可逆网络都使用恒定的内存来计算梯度。

其中最主要目的就是为了减少内存的消耗,当前所有的神经网络都采用反向传播的方式来训练,反向传播算法需要存储网络的中间结果来计算梯度,而且其对内存的消耗与网络单元数成正比。这也就意味着,网络越深越广,对内存的消耗越大,这将成为很多应用的瓶颈。

下面是 Pytorch summary 的结果,Forward/backward pass size(MB): 218.59 就是需要保存的中间变量大小,可以看出这部分占据了很大部分显存(随着网络深度的增加,中间变量占据显存量会一直增加,resnet152(size=224)的中间变量更是占据总共内存的 606.6÷836.79≈0.725 )。如果不存储中间层结果,那么就可以大幅减少 GPU 的显存占用,有助于训练更深更广的网络。

import torch from torchvision import models from torchsummary import summary device = torch.device('cuda' if torch.cuda.is_available() else 'cpu') vgg = models.vgg16().to(device) summary(vgg, (3, 224, 224))

结果:

---------------------------------------------------------------- Layer (type) Output Shape Param # ================================================================ Conv2d-1 [-1, 64, 224, 224] 1,792 ReLU-2 [-1, 64, 224, 224] 0 Conv2d-3 [-1, 64, 224, 224] 36,928 ReLU-4 [-1, 64, 224, 224] 0 MaxPool2d-5 [-1, 64, 112, 112] 0 Conv2d-6 [-1, 128, 112, 112] 73,856 ReLU-7 [-1, 128, 112, 112] 0 Conv2d-8 [-1, 128, 112, 112] 147,584 ReLU-9 [-1, 128, 112, 112] 0 MaxPool2d-10 [-1, 128, 56, 56] 0 Conv2d-11 [-1, 256, 56, 56] 295,168 ReLU-12 [-1, 256, 56, 56] 0 Conv2d-13 [-1, 256, 56, 56] 590,080 ReLU-14 [-1, 256, 56, 56] 0 Conv2d-15 [-1, 256, 56, 56] 590,080 ReLU-16 [-1, 256, 56, 56] 0 MaxPool2d-17 [-1, 256, 28, 28] 0 Conv2d-18 [-1, 512, 28, 28] 1,180,160 ReLU-19 [-1, 512, 28, 28] 0 Conv2d-20 [-1, 512, 28, 28] 2,359,808 ReLU-21 [-1, 512, 28, 28] 0 Conv2d-22 [-1, 512, 28, 28] 2,359,808 ReLU-23 [-1, 512, 28, 28] 0 MaxPool2d-24 [-1, 512, 14, 14] 0 Conv2d-25 [-1, 512, 14, 14] 2,359,808 ReLU-26 [-1, 512, 14, 14] 0 Conv2d-27 [-1, 512, 14, 14] 2,359,808 ReLU-28 [-1, 512, 14, 14] 0 Conv2d-29 [-1, 512, 14, 14] 2,359,808 ReLU-30 [-1, 512, 14, 14] 0 MaxPool2d-31 [-1, 512, 7, 7] 0 Linear-32 [-1, 4096] 102,764,544 ReLU-33 [-1, 4096] 0 Dropout-34 [-1, 4096] 0 Linear-35 [-1, 4096] 16,781,312 ReLU-36 [-1, 4096] 0 Dropout-37 [-1, 4096] 0 Linear-38 [-1, 1000] 4,097,000 ================================================================ Total Params: 138,357,544 Trainable params: 138,357,544 Non-trainable params: 0 ---------------------------------------------------------------- Input size (MB): 0.57 Forward/backward pass size (MB): 218.59 Params size (MB): 527.79 Estimated Total Size (MB): 746.96 ----------------------------------------------------------------

接下来我将先从可逆神经网络讲起,然后是神经网络的反向传播,最后是标准残差网络。对反向传播算法和标准残差网络比较熟悉的小伙伴,可以只看第一节:可逆神经网络。如果各位小伙伴不熟悉反向传播算法和标准残差网络,建议先看第二节:反向传播(BP)算法和第三节:残差网络(Residual Network)。本文1.2和1.3.4摘录自 @阿亮。

可逆神经网络

可逆网络具有的性质:

  1. 网络的输入、输出的大小必须一致。
  2. 网络的雅可比行列式不为 0。
1.1 什么是雅可比行列式?

雅可比行列式通常称为雅可比式(Jacobian),它是以 n 个 n 元函数的偏导数为元素的行列式 。事实上,在函数都连续可微(即偏导数都连续)的前提之下,它就是函数组的微分形式下的系数矩阵(即雅可比矩阵)的行列式。若因变量对自变量连续可微,而自变量对新变量连续可微,则因变量也对新变量连续可微。这可用行列式的乘法法则和偏导数的连锁法则直接验证。也类似于导数的连锁法则。偏导数的连锁法则也有类似的公式;这常用于重积分的计算中。

神经网络发展历程简单介绍(可逆神经网络详细解析)(2)

神经网络发展历程简单介绍(可逆神经网络详细解析)(3)

神经网络发展历程简单介绍(可逆神经网络详细解析)(4)

1.2 雅可比行列式与神经网络的关系

为什么神经网络会与雅可比行列式有关系?这里我借用李宏毅老师的 ppt(12-14页)。想看视频的可以到 b 站上看。

神经网络发展历程简单介绍(可逆神经网络详细解析)(5)

神经网络发展历程简单介绍(可逆神经网络详细解析)(6)

神经网络发展历程简单介绍(可逆神经网络详细解析)(7)

简单的来讲就是 ,他们的分布之间的关系就变为 ,又因为有 ,所以 这个网络的雅可比行列式不为 0 才行。

顺便提一下,flow-based Model 优化的损失函数如下:

神经网络发展历程简单介绍(可逆神经网络详细解析)(8)

其实这里跟矩阵运算很像,矩阵可逆的条件也是矩阵的雅可比行列式不为 0,雅可比矩阵可以理解为矩阵的一阶导数。

假设可逆网络的表达式为:

神经网络发展历程简单介绍(可逆神经网络详细解析)(9)

神经网络发展历程简单介绍(可逆神经网络详细解析)(10)

它的雅可比矩阵为:

神经网络发展历程简单介绍(可逆神经网络详细解析)(11)

其行列式为 1。

1.3 可逆残差网络(Reversible Residual Network)

神经网络发展历程简单介绍(可逆神经网络详细解析)(12)

论文标题:The Reversible Residual Network: Backpropagation Without Storing Activations论文链接:https://arxiv.org/abs/1707.04585多伦多大学的 Aidan N.Gomez 和 Mengye Ren 提出了可逆残差神经网络,当前层的激活结果可由下一层的结果计算得出,也就是如果我们知道网络层最后的结果,就可以反推前面每一层的中间结果。这样我们只需要存储网络的参数和最后一层的结果即可,激活结果的存储与网络的深度无关了,将大幅减少显存占用。令人惊讶的是,实验结果显示,可逆残差网络的表现并没有显著下降,与之前的标准残差网络实验结果基本旗鼓相当。

1.3.1 可逆块结构

可逆神经网络将每一层分割成两部分,分别为 和 ,每一个可逆块的输入是 ,输出是 。其结构如下:

正向计算图示:

神经网络发展历程简单介绍(可逆神经网络详细解析)(13)

公式表示:

神经网络发展历程简单介绍(可逆神经网络详细解析)(14)

逆向计算图示:

神经网络发展历程简单介绍(可逆神经网络详细解析)(15)

公式表示:

神经网络发展历程简单介绍(可逆神经网络详细解析)(16)

其中 F 和 G 都是相似的残差函数,参考上图残差网络。可逆块的跨距只能为 1,也就是说可逆块必须一个接一个连接,中间不能采用其它网络形式衔接,否则的话就会丢失信息,并且无法可逆计算了,这点与残差块不一样。如果一定要采取跟残差块相似的结构,也就是中间一部分采用普通网络形式衔接,那中间这部分的激活结果就必须显式的存起来。

1.3.2 不用存储激活结果的反向传播

为了更好地计算反向传播的步骤,我们修改一下上述正向计算和逆向计算的公式:

神经网络发展历程简单介绍(可逆神经网络详细解析)(17)

尽管 和 的值是相同的,但是两个变量在图中却代表不同的节点,所以在反向传播中它们的总体导数是不一样的。 的导数包含通过 产生的间接影响,而 的导数却不受 的任何影响。

在反向传播计算流程中,先给出最后一层的激活值 和误差传播的总体导数 ,然后要计算出其输入值 和对应的导数 ,以及残差函数 F 和 G 中权重参数的总体导数,求解步骤如下:

神经网络发展历程简单介绍(可逆神经网络详细解析)(18)

1.3.3 计算开销

一个 N 个连接的神经网络,正向计算的理论加乘开销为 N,反向传播求导的理论加乘开销为 2N(反向求导包含复合函数求导连乘),而可逆网络多一步需要反向计算输入值的操作,所以理论计算开销为 4N,比普通网络开销约多出 33% 左右。但是在实际操作中,正向和反向的计算开销在 GPU 上差不多,可以都理解为 N。那么这样的话,普通网络的整体计算开销为 2N,可逆网络的整体开销为 3N,也就是多出了约 50%。

1.3.4 雅可比行列式的计算

神经网络发展历程简单介绍(可逆神经网络详细解析)(19)

编码公式如下:

神经网络发展历程简单介绍(可逆神经网络详细解析)(20)

神经网络发展历程简单介绍(可逆神经网络详细解析)(21)

解码公式如下:

神经网络发展历程简单介绍(可逆神经网络详细解析)(22)

神经网络发展历程简单介绍(可逆神经网络详细解析)(23)

为了计算雅可比矩阵,我们更直观的写成下面的编码公式:

神经网络发展历程简单介绍(可逆神经网络详细解析)(24)

神经网络发展历程简单介绍(可逆神经网络详细解析)(25)

它的雅可比矩阵为:

神经网络发展历程简单介绍(可逆神经网络详细解析)(26)

其实上面这个雅可比行列式也是 1,因为这里 ,它们的系数是一样的。

有另外一种解释方式就是把这种对偶的形式切成两半:

神经网络发展历程简单介绍(可逆神经网络详细解析)(27)

神经网络发展历程简单介绍(可逆神经网络详细解析)(28)

神经网络发展历程简单介绍(可逆神经网络详细解析)(29)

其行列式为 1。

神经网络发展历程简单介绍(可逆神经网络详细解析)(30)

神经网络发展历程简单介绍(可逆神经网络详细解析)(31)

因为是对偶的形式,所以这里的行列式也为 1。

因为 ,所以其行列式也为 1。

反向传播(BP)算法

神经网络发展历程简单介绍(可逆神经网络详细解析)(32)

上图中符号的含义:

正向传播计算过程:

神经网络发展历程简单介绍(可逆神经网络详细解析)(33)

神经网络发展历程简单介绍(可逆神经网络详细解析)(34)

反向传播计算过程:

以单个样本为例,假设输入向量是 [x1,x2,x3],目标输出值是 [y1,y2],代价函数用 L 表示。反向传播的总体原理就是根据总体输出误差,反向传播回网络,通过计算每一层节点的梯度,利用梯度下降法原理,更新每一层的网络权重 w 和偏置 b,这也是网络学习的过程。误差反向传播的优点就是可以把繁杂的导数计算以数列递推的形式来表示, 简化了计算过程。

以平方误差来计算反向传播的过程,代价函数表示如下:

神经网络发展历程简单介绍(可逆神经网络详细解析)(35)

根据导数的链式法则反向求解隐藏 -> 输出层、输入层 -> 隐藏层的权重表示:

神经网络发展历程简单介绍(可逆神经网络详细解析)(36)

引入新的误差求导表示形式,称为神经单元误差:

神经网络发展历程简单介绍(可逆神经网络详细解析)(37)

l=2,3 表示第几层,j 表示某一层的第几个节点。替换表示后如下:

神经网络发展历程简单介绍(可逆神经网络详细解析)(38)

所以我们可以归纳出一般的计算公式:

神经网络发展历程简单介绍(可逆神经网络详细解析)(39)

从上述公式可以看出,如果神经单元误差 δ 可以求出来,那么总误差对每一层的权重 w 和偏置 b 的偏导数就可以求出来,接下来就可以利用梯度下降法来优化参数了。

求解每一层的 δ:

神经网络发展历程简单介绍(可逆神经网络详细解析)(40)

神经网络发展历程简单介绍(可逆神经网络详细解析)(41)

也就是说,我们根据输出层的神经误差单元 δ 就可以直接求出隐藏层的神经误差单元,进而省去了隐藏层的繁杂的求导过程,我们可以得出更一般的计算过程:

神经网络发展历程简单介绍(可逆神经网络详细解析)(42)

从而得出 l 层神经单元误差和 l 1 层神经单元误差的关系。这就是误差反向传播算法,只要求出输出层的神经单元误差,其它层的神经单元误差就不需要计算偏导数了,而可以直接通过上述公式得出。

残差网络(Residual Network)

残差网络主要可以解决两个问题(其结构如下图):

神经网络发展历程简单介绍(可逆神经网络详细解析)(43)

上述结构就是一个两层网络组成的残差块,残差块可以由 2、3 层甚至更多层组成,但是如果是一层的,就变成线性变换了,没什么意义了。上述图可以写成公式如下:

神经网络发展历程简单介绍(可逆神经网络详细解析)(44)

所以在第二层进入激活函数 ReLU之 前 F(x) x 组成新的输入,也叫恒等映射。

恒等映射就是在这个残差块输入是 x 的情况下输出依然是 x,这样其目标就是学习让 F(X)=0。

这里有一个问题哈,为什么要额外加一个 x 呢,而不是让模型直接学习 F(x)=x?

因为让 F(x)=0 比较容易,初始化参数 W 非常小接近 0,就可以让输出接近 0,同时输出如果是负数,经过第一层 Relu 后输出依然 0,都能使得最后的 F(x)=0,也就是有多种情况都可以使得 F(x)=0;但是让 F(x)=x 确实非常难的,因为参数都必须刚刚好才能使得最后输出为 x。

恒等映射有什么作用?

恒等映射就可以解决网络退化的问题,当网络层数越来越深的时候,网络的精度却在下降,也就是说网络自身存在一个最优的层度结构,太深太浅都能使得模型精度下降。有了恒等映射存在,网络就能够自己学习到哪些层是冗余的,就可以无损通过这些层,理论上讲再深的网络都不影响其精度,解决了网络退化问题。

为什么可以解决梯度消失问题呢?

以两个残差块的结构实例图来分析,其中每个残差块有 2 层神经网络组成,如下图:

神经网络发展历程简单介绍(可逆神经网络详细解析)(45)

假设激活函数 ReLU 用 g(x) 函数来表示,样本实例是 [x1,y1],即输入是 x1,目标值是 y1,损失函数还是采用平方损失函数,则每一层的计算如下:

神经网络发展历程简单介绍(可逆神经网络详细解析)(46)

下面我们对第一个残差块的权重参数求导,根据链式求导法则,公式如下:

神经网络发展历程简单介绍(可逆神经网络详细解析)(47)

我们可以看到求导公式中多了一个 1项,这就将原来的链式求导中的连乘变成了连加状态,可以有效避免梯度消失了

参考文献:

[1] PPT https://speech.ee.ntu.edu.tw/~tlkagk/courses/ML_2019/Lecture/FLOW (v7).pdf

[2] 神经网络的可逆形式 https://zhuanlan.zhihu.com/p/268242678

[3] 大幅减少GPU显存占用:可逆残差网络(The Reversible Residual Network) https://www.cnblogs.com/gczr/p/12181354.html

[4] 雅可比行列式 https://baike.baidu.com/item/雅可比行列式/4709261?fr=aladdin

[5] The Reversible Residual Network: Backpropagation Without Storing Activations

[6] pytorch-summary https://github.com/sksq96/pytorch-summary

,