7 AdaGrad算法
7.1 稀疏特征和学习率
假设我们正在训练一个语言模型。为了获得良好的准确性,我们大多希望在训练的过程中降低学习率,速度通常为$O^{(t−\frac{1}{2})}$或更低。
现在讨论关于稀疏特征(即只在偶尔出现的特征)的模型训练,这对自然语言来说很常见。例如,我们看到“预先条件”这个词比“学习”这个词的可能性要小得多。但是,它在计算广告学和个性化协同过滤等其他领域也很常见。只有在这些不常见的特征出现时,与其相关的参数才会得到有意义的更新。鉴于学习率下降,我们可能最终会面临这样的情况:常见特征的参数相当迅速地收敛到最佳值,而对于不常见的特征,我们仍缺乏足够的观测以确定其最佳值。换句话说,学习率要么对于常见特征而言降低太慢,要么对于不常见特征而言降低太快。
解决此问题的一个方法是记录我们看到特定特征的次数,然后将其用作调整学习率。即我们可以使用大小为$\eta_ i = \frac{ \eta_ 0}{ \sqrt{ s(i, t) + c}}$的学习率,而不是$\eta = \frac{ \eta_ 0 }{ \sqrt{ t + c}}$,其中$s(i, t)$是计下了我们截至t时观察到特征i的次数.
AdaGrad算法通过将粗略的计数器s(i,t)替换为先前观察所得梯度的平方之和来解决这个问题。它使用$s(i,t+1)=s(i,t)+(\partial_ i f(x))^2$来调整学习率。这有两个好处:首先,我们不再需要决定梯度何时算足够大。其次,它会随梯度的大小自动变化。通常对应于较大梯度的坐标会显著缩小,而其他梯度较小的坐标则会得到更平滑的处理。在实际应用中,它促成了计算广告学及其相关问题中非常有效的优化程序。但是,它遮盖了AdaGrad固有的一些额外优势,这些优势在预处理环境中很容易被理解。
7.2 预处理
凸优化问题有助于分析算法的特点。毕竟对大多数非凸问题来说,获得有意义的理论保证很难,但是直觉和洞察往往会延续。让我们来看看最小化$f(\mathbf{x})=\frac12\mathbf{x}^\top\mathbf{Q}\mathbf{x}+\mathbf{c}^\top\mathbf{x}+b$这一问题。
同上一节, 我们可以根据其特征分解$\mathbf{ Q }=\mathbf{ U }^⊤\mathbf{ Λ }\mathbf{ U }$重写这个问题,来得到一个简化得多的问题,使每个坐标都可以单独解出:
$$f(\mathbf{x})=\bar{f}(\bar{\mathbf{x}})=\frac12\bar{\mathbf{x}}^\top\boldsymbol{\Lambda}\bar{\mathbf{x}}+\bar{\mathbf{c}}^\top\bar{\mathbf{x}}+b.$$- 其中x=Ux,且因此c=Uc。修改后优化器为$\bar{ \mathbf { x } }= -\boldsymbol{ \Lambda }^{-1}\bar{ \mathbf { c } }$。且最小值为:$-\frac{1}{2}\bar{ \mathbf { c } }^\top\boldsymbol{ \Lambda }^{-1}\bar{ \mathbf { c } }+b$。这样更容易计算,因为Λ是一个包含Q特征值的对角矩阵。
如果稍微扰动c,我们会期望在f的最小化器中只产生微小的变化。遗憾的是,情况并非如此。虽然c的微小变化导致了$\bar c$同样的微小变化,但f的(以及$\bar f$的)最小化器并非如此。每当特征值Λi很大时,我们会看到$\bar{x_i}$和$\bar f$的最小值发声微小变化。相反,对小的Λi来说,$\bar x_i$的变化可能是剧烈的。最大和最小的特征值之比称为优化问题的条件数
$$ k = \frac{ \mathbf{ \Lambda }{ 1 } }{ \mathbf{ \Lambda }{ d } }.$$- 如果条件数k很大, 准确解决优化问题就会很难。我们需要确保在获取大量动态的特征值范围时足够谨慎:难道我们不能简单地通过扭曲空间来“修复”这个问题,从而使所有特征值都是1?理论上这很容易:我们只需要Q的特征值和特征向量即可将问题从x整理到$z := \Lambda^ \frac12 \mathbf{ U } \mathbf{ x }$中的一个。在新的坐标系中,$x^⊤Qx$可以被简化为$||z||^2$.可惜,这是一个相当不切实际的想法。一般而言,计算特征值和特征向量要比解决实际问题“贵”得多。
虽然准确计算特征值可能会很昂贵,但即便只是大致猜测并计算它们,也可能已经比不做任何事情好得多。特别是,我们可以使用Q的对角线条目并相应地重新缩放它。这比计算特征值开销小的多。
$$\tilde{ \mathbf{ Q } } = diag^{-\frac12}(\mathbf{ Q })\mathbf{ Q } diag^{-\frac12}(\mathbf{ Q })$$其中$\tilde{ \mathbf{ Q } }_ {ij} = \frac{ \mathbf{ Q }{ij} }{ \sqrt{ \mathbf{ Q }{ii} \mathbf{ Q }{jj} } }$。特别注意对于所有i,$\tilde{ \mathbf{ Q } } {ii} = 1$。在大多数情况下,这大大简化了条件数。例如我们之前讨论的案例,它将完全消除眼下的问题,因为问题是轴对齐的。
遗憾的是,我们还面临另一个问题:在深度学习中,我们通常情况甚至无法计算目标函数的二阶导数:对于$\mathbf x \in \mathbb R^{d}$,即使只在小批量上,二阶导数可能也需要$O(d^2)$空间来计算,导致几乎不可行。AdaGrad算法巧妙的思路是,使用一个代理来表示Hessian矩阵的对角线,既相对易于计算又高效。为了了解它是如何生效的,让我们来看看$\bar f(\bar x)$。我们有:
$$\partial_{\bar{\mathbf{x}}}\bar{f}(\bar{\mathbf{x}})=\boldsymbol{\Lambda}\bar{\mathbf{x}}+\bar{\mathbf{c}}=\boldsymbol{\Lambda}\left(\bar{\mathbf{x}}-\bar{\mathbf{x}}_0\right)$$
- 其中$\bar x_0$是$\bar f$的优化器。因此,梯度的大小取决于Λ和与最佳值的差值。如果$\bar x− \bar x_0$没有改变,那这就是我们所求的。毕竟在这种情况下,梯度$\partial_ \bar x \bar f(\bar x)的大小就足够了。由于AdaGrad算法是一种随机梯度下降算法,所以即使是在最佳值中,我们也会看到具有非零方差的梯度。因此,我们可以放心地使用梯度的方差作为黑塞矩阵比例的廉价替代。
7.3 算法
使用变量$\mathbf{ s }_ t$来累加过去的梯度方差:
$$\mathbf{ g }t = \partial \mathbf w l( y_t , f( \mathbf x_t , \mathbf w ))$$
$$\mathbf{ s }t = \mathbf{ s }{ t-1 } + \mathbf{ g }t^2$$
$$\mathbf{ w }{ t } = \mathbf{ w }_{t-1} - \frac{ \eta }{ \sqrt{ \mathbf{ s }_t + \epsilon } } \cdot \mathbf{ g }_t$$在这里,操作是按照坐标顺序应用。也就是说,$v^ 2$有条目$v^2 _ i$。同样, $\frac{1}{\sqrt{v}}$有条目 $\frac{1}{\sqrt{v_i}}$,并且u·v有条目$u_i v_i$。初始化s0 =0
就像在动量法中我们需要跟踪一个辅助变量一样,在AdaGrad算法中,我们允许每个坐标有单独的学习率。与SGD算法相比,这并没有明显增加AdaGrad的计算代价,因为主要计算用在$l(y_t,f(x_t,w))$及其导数。在st中累加平方梯度意味着st基本上以线性速率增长(由于梯度从最初开始衰减,实际上比线性慢一些)。这产生了一个学习率$O{t^{−\frac{1}{2}}}$,但是在单个坐标的层面上进行了调整。对于凸问题,这完全足够了。
看看他在二次凸函数中的表现如何:
$$ f( \mathbf x ) = 0.1x_1^2 + 2x_2^2$$
1 | %matplotlib inline |
epoch 20, x1: -2.382563, x2: -0.158591
c:\Users\admin\miniconda3\envs\d2l\lib\site-packages\torch\functional.py:478: UserWarning: torch.meshgrid: in an upcoming release, it will be required to pass the indexing argument. (Triggered internally at C:\actions-runner\_work\pytorch\pytorch\builder\windows\pytorch\aten\src\ATen\native\TensorShape.cpp:2895.)
return _VF.meshgrid(tensors, **kwargs) # type: ignore[attr-defined]
使用与之前相同的学习率来实现AdaGrad算法,即η=0.4。可以看到,自变量的迭代轨迹较平滑。但由于st的累加效果使学习率不断衰减,自变量在迭代后期的移动幅度较小。
将学习率提高到2,可以看到更好的表现。这已经表明,即使在无噪声的情况下,学习率的降低可能相当剧烈,我们需要确保参数能够适当地收敛。
1 | eta = 2 |
epoch 20, x1: -0.002295, x2: -0.000000
7.4 从零实现
1 | def init_adagrad_states(feature_dim): |
loss: 0.243, 0.009 sec/epoch
7.5 简介实现
1 | trainer = torch.optim.Adagrad |
loss: 0.242, 0.009 sec/epoch