用时间换取效果:Keras梯度累积优化器
By 苏剑林 | 2019-07-08 | 80248位读者 |现在Keras中你也可以用小的batch size实现大batch size的效果了——只要你愿意花$n$倍的时间,可以达到$n$倍batch size的效果,而不需要增加显存。
Github地址:https://github.com/bojone/accum_optimizer_for_keras
扯淡 #
在一两年之前,做NLP任务都不用怎么担心OOM问题,因为相比CV领域的模型,其实大多数NLP模型都是很浅的,极少会显存不足。幸运或者不幸的是,Bert出世了,然后火了。Bert及其后来者们(GPT-2、XLNET等)都是以足够庞大的Transformer模型为基础,通过足够多的语料预训练模型,然后通过fine tune的方式来完成特定的NLP任务。
即使你很不想用Bert,但现在的实际情况是:你精心设计的复杂的模型,效果可能还不如简单地fine tune一下Bert好。所以不管怎样,为了跟上时代,总得需要学习一下Bert的fine tune。问题是“不学不知道,一学吓一跳”,只要任务稍微复杂一点,或者句子长度稍微长一点,显存就不够用了,batch size急剧下降——32?16?8?一跌再跌都是有可能的。
这不难理解,Transformer基于Attention,而Attention理论上空间和时间复杂度都是$\mathcal{O}(n^2)$,虽然在算力足够强的时候,Attention由于其并行性还是可以表现得足够快,但是显存占用量是省不了了,$\mathcal{O}(n^2)$意味着当你句子长度变成原来的2倍时,显存占用基本上就需要原来的4倍,这个增长比例肯定就容易OOM了~
而更不幸的消息是,大家都在fine tune预训练Bert的情况下,你batch_size=8可能比别人batch_size=80低好几个千分点甚至是几个百分点,显然这对于要刷榜的读者是很难受的。难道除了加显卡就没有别的办法了吗?
正事 #
有!通过梯度缓存和累积的方式,用时间来换取空间,最终训练效果等效于更大的batch size。因此,只要你跑得起batch_size=1,只要你愿意花$n$倍的时间,就可以跑出$n$倍的batch size了。
梯度累积的思路,在之前的文章《“让Keras更酷一些!”:小众的自定义优化器》已经介绍了,当时称之为“软batch(soft batch)”,本文还是沿着主流的叫法称之为“梯度累积(accumulate gradients)”好了。所谓梯度累积,其实很简单,我们梯度下降所用的梯度,实际上是多个样本算出来的梯度的平均值,以batch_size=128为例,你可以一次性算出128个样本的梯度然后平均,我也可以每次算16个样本的平均梯度,然后缓存累加起来,算够了8次之后,然后把总梯度除以8,然后才执行参数更新。当然,必须累积到了8次之后,用8次的平均梯度才去更新参数,不能每算16个就去更新一次,不然就是batch_size=16了。
刚才说了,在之前的文章的那个写法是有误的,因为用到了
K.switch(cond, K.update(p, new_p), p)
来控制更新,但事实上这个写法不能控制更新,因为K.switch
只保证结果的选择性,不保证执行的选择性,事实上它等价于
cond * K.update(p, new_p) + (1 - cond) * p
也就是说不管cond
如何,两个分支都是被执行了。事实上Keras或Tensorflow“几乎”不存在只执行一个分支的条件写法(说“几乎”是因为在一些比较苛刻的条件下可以做到),所以此路不通。
不能这样写的话,那只能在“更新量”上面下功夫,如前面所言,每次算16个样本的梯度,每次都更新参数,只不过8次中有7次的更新量是0,而只有1次是真正的梯度下降更新。很幸运的是,这种写法还可以无缝地接入到现有的Keras优化器中,使得我们不需要重写优化器!详细写法请看:
具体的写法无外乎就是一些移花接木的编程技巧,真正有技术含量的部分不多。关于写法本身不再细讲,如果有疑问欢迎讨论区讨论。
(注:这个优化器的修改,使得小batch size能起到大batch size的效果,前提是模型不包含Batch Normalization,因为Batch Normalization在梯度下降的时候必须用整个batch的均值方差。所以如果你的网络用到了Batch Normalization,想要准确达到大batch size的效果,目前唯一的方法就是加显存/加显卡。)
实验 #
至于用法则很简单:
opt = AccumOptimizer(Adam(), 10) # 10是累积步数
model.compile(loss='mse', optimizer=opt)
model.fit(x_train, y_train, epochs=10, batch_size=10)
这样一来就等价于batch_size=100的Adam优化器了,代价就是你每个epoch的速度会更慢了(因为batch size更小了),好处是你只需要用到batch_size=10的显存量。
可能读者想问的一个问题是:你怎么证明你的写法生效了?也就是说你怎么证明你的结果确实是batch_size=100而不是batch_size=10?为此,我做了个比较极端的实验,代码在这里:
https://github.com/bojone/accum_optimizer_for_keras/blob/master/mnist_mlp_example.py
代码很简单,就是用多层MLP做MNIST分类,用Adam优化器,fit
的时候batch_size=1。优化器有两个选择,第一个是直接Adam()
,第二个是AccumOptimizer(Adam(), 100)
:
如果是直接Adam(),那loss一直在0.4上下徘徊,后面loss越来越大了(训练集都这样),val的准确率也没超过97%;
如果是AccumOptimizer(Adam(), 100),那么训练集的loss越来越低,最终降到0.02左右,val的最高准确率有98%+;
最后我比较了直接Adam()但是batch_size=100的结果,发现跟AccumOptimizer(Adam(), 100)但是batch_size=1时表现差不多。
这个结果足以表明写法生效了,达到了预期的目的。如果这还不够说服力,我再提供一个训练结果作为参考:在某个Bert的fine tune实验中,直接用Adam()加batch_size=12,我跑到了70.33%的准确率;我用AccumOptimizer(Adam(), 10)加batch_size=12(预期等效batch size是120),我跑到了71%的准确率,提高了0.7%,如果你在刷榜,那么这0.7%可能是决定性的。
结论 #
终于把梯度累积(软batch)正式地实现了,以后用Bert的时候,也可以考虑用大batch_size了哈~
转载到请包括本文地址:https://spaces.ac.cn/archives/6794
更详细的转载事宜请参考:《科学空间FAQ》
如果您还有什么疑惑或建议,欢迎在下方评论区继续讨论。
如果您觉得本文还不错,欢迎分享/打赏本文。打赏并非要从中获得收益,而是希望知道科学空间获得了多少读者的真心关注。当然,如果你无视它,也不会影响你的阅读。再次表示欢迎和感谢!
如果您需要引用本文,请参考:
苏剑林. (Jul. 08, 2019). 《用时间换取效果:Keras梯度累积优化器 》[Blog post]. Retrieved from https://spaces.ac.cn/archives/6794
@online{kexuefm-6794,
title={用时间换取效果:Keras梯度累积优化器},
author={苏剑林},
year={2019},
month={Jul},
url={\url{https://spaces.ac.cn/archives/6794}},
}
July 10th, 2019
从阅读理解比赛就一直开始关注苏神了,想请教下keras如何写模型的ema,自己实现的感觉并没有提升
https://kexue.fm/archives/6575
July 23rd, 2019
# 累积梯度 (gradient accumulation)
52 => self.accum_grads = [K.zeros(K.int_shape(p), dtype=K.dtype(p)) for p in params]
grads = self.get_gradients(loss, params)
for g, ag in zip(grads, self.accum_grads):
self.updates.append(K.update(ag, K.switch(self.cond, ag * 0, ag + g)))
第52行每次get_updates时都先清空accum_grads,不是没法实现累积了吗?请苏神赐教。
那是你不理解tf这种静态图的写法和逻辑。get_updates从头到尾只会执行一次,用来构建整个计算图(的一部分)。
July 30th, 2019
累积梯度在cv里也就是对下降可能有点儿帮助,网络中如果有别的统计量(比如bn层的moving_mean和moving_var)没办法通过这种方式受益。另外博主真的博学啊,看到这个网站都是一个人更新,惊呆了。
谢谢,过奖了。
严谨起见,我原来就已经在文中声明了对BN无效了。
September 21st, 2019
苏神我想问一哈,您写的ema可以和这个一起使用吗
可以,但是ema的momentum需要加大。
October 12th, 2019
苏神,我想请教您一下,我在使用bert配合您这个梯度累积方法的时候,我的输出层的shape(seq_len, 768)时会报错,有什么办法修改一下吗...
已解决,太粗心了。
June 19th, 2020
感谢!找了好久,终于找到啦~
July 9th, 2020
其实,tf.keras v2版是可以分支执行的, 这样逻辑更清晰,代码更简洁。
def extend_with_gradient_accumulation_v2()可修改如下:
我对tf2没什么兴趣~
哈哈,,老哥好刚,,确实,这个静态图没很好办法实现这个逻辑
tf.keras v2 不可以分支执行
摘自源码:
**WARNING**: Any Tensors or Operations created outside of `true_fn` and
`false_fn` will be executed regardless of which branch is selected at runtime.
请问这个是否可以用
June 28th, 2021
苏神,在调试代码的时候一直报这个错:
TypeError: __array__() takes 1 positional argument but 2 were given
尝试了几种方式后没有解决,报错位置是在:
self.optimizer.lr = K.switch(self.cond, self.optimizer.lr, 0.)
可以帮忙看下吗?我的是tf2.4
September 30th, 2021
EM算法或交替算法,,,,比如第一个batch,A部分不变B网络更新;第二个batch,A更新B不变。。这种在静态图,比如TF1,keras都是怎么实现?
这不就是像GAN一样吗,参考任意一个GAN实现就行了,tf1、keras都可以实现GAN
October 22nd, 2021
话说苏神,用梯度累积的话,因为改变了batchsize,是不是也要按照线性策略修改学习率?
你的意思是batch_size扩大多少倍,学习率也扩大多少倍?这个在一定范围内可以这样做,但是如果batch_size扩大了几百倍,学习率总不能也扩大几百倍吧?
比如原始batch_size = 8 , lr = 0.1
现在用了梯度累积方法,假设累积步长为4,那么视为新的等效batch_size = 32,我的已问就是此时是否需要将学习率改为0.4呢?
我上面说了呀,有限范围内可以这样做,但是学习率本身太大就不行了。比如你原始lr=0.0001,那么batch_size变为4倍后,将lr设为0.0004基本上没啥问题;但如果你原始lr=0.1,那么就算batch_size变为4倍,改为lr=0.4很可能也不收敛了。