![PyTorch深度学习应用实战](https://wfqqreader-1252317822.image.myqcloud.com/cover/410/52842410/b_52842410.jpg)
3-3 自动微分
反向传导时,会更新每一层的权重,这时就轮到偏微分运算派上用场,所以,深度学习框架的第二项主要功能就是自动微分(Automatic Differentiation)。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P44_25830.jpg?sign=1739274621-C0c7ULqyRogiDQRcmZPT6jm3CpK795Qa-0-6ddbe66494eefc5fbb8d8854243cd31d)
图3.9 神经网络权重求解过程
下列程序代码请参考【03_02_自动微分.ipynb】。
(1)变量y会对x自动微分,代码如下。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P45_3069.jpg?sign=1739274621-w0ogVVmZH4Ec9yqxeWd9tySst9b88L7i-0-225fd811d25d6c4ffdfb91cc815ee3d7)
变量x要参与自动微分,须指定参数requires_grad=True。
设定y是x的多项式,y.grad_fn可取得y的梯度函数。
执行y.backward(),会进行反向传导,即偏微分。
通过x.grad可取得梯度,若有多个变量也是如此。
执行结果中x^2对x自动偏微分,得2x,因x=4,故x梯度=8。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P45_3083.jpg?sign=1739274621-oIm1mg0jp2HxZCD1jYet3fI0X1l5hxFx-0-34f7f988a16b524efafac6ff7e6daa46)
(2)下列程序代码可取得自动微分相关的属性值。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P45_3086.jpg?sign=1739274621-2Y10t1IM2B9fBd05rj7DEy1FGmEpQ515-0-f117b4f38188d5eea8eb386a3450b83c)
执行结果。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P45_3093.jpg?sign=1739274621-vA9GTGrdg6NCFU4bJ6GSmOkrTH5vPB29-0-38a35ac6fdbb7beda478432f981b7e6d)
(3)来看一个较复杂的例子,以神经网络进行分类时,常使用交叉熵(Cross Entropy)作为损失函数,下图表达Cross Entropy=CE(y, wx+b)。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_25835.jpg?sign=1739274621-2OURaOOzOjhLPhGEDOrIWSXkhbDWsqzH-0-d6f677a9caa7b84f63f859e2db9dbf7f)
图3.10 交叉熵(Cross Entropy)运算图
PyTorch会依据程序,建构运算图(Computational Graph),描述梯度下降时,变量运算的顺序如下所示,先算x,再算z,最后计算loss。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_3165.jpg?sign=1739274621-V9sTOqA3nTHaAsPlRreUUzX46eBi8Czd-0-7d12b14ea3cc6ddbfa6130fa02aa7831)
torch.nn.functional.binary_cross_entropy_with_logits是PyTorch提供的交叉熵函数。
执行结果:
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_3172.jpg?sign=1739274621-I55d3XTOTaynCtHLopEnVXG9SAJEMwmK-0-765f0ba1e67821271c94ab28fd0e7f7d)
(4)自动微分中,z是w、b的函数,而loss又是z的函数,故只要对loss进行反向传导即可。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_3175.jpg?sign=1739274621-PilPBdUUqu1a3swoSu0IAkptJYGyynBH-0-698ade47a1488c70835f9d0b64fbb042)
执行结果中,w、b梯度分别为:
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_3182.jpg?sign=1739274621-HcBq7dVN2WQQEpIboJGFA8oRYnAgjekJ-0-b64955a1987adedd83492e8fa57bc6db)
(5)TensorFlow使用常数(Constant)及变量(Variable),而PyTorch自v0.4.0起已弃用Variable,直接使用tensor即可,但网络上依然常见Variable,特此说明,详情请参阅本章参考文献[1]。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_3185.jpg?sign=1739274621-HfPQub8MZhiStFWMR0iVLGg32dlIYexS-0-ab1771f81500e79eba673bd6802083ce)
上例以torch.ones替代Variable。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_3192.jpg?sign=1739274621-Bz0WdmnkZ4OOsuitMd1TqUpB2AEEMrx4-0-7967453f9cd4d0e968ce5677774ea90b)
(6)模型训练时,会反复执行正向/反向传导,以找到最佳解,因此,梯度下降会执行很多次,这时要注意两件事:
· y.backward执行后,预设会将运算图销毁,y.backward将无法再执行,故要保留运算图,须加参数retain_graph=True。
· 梯度会不断累加,因此,执行y.backward后要重置(Reset)梯度,指令如下:x.grad.zero_()
(7)不重置梯度代码如下:
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P47_3214.jpg?sign=1739274621-oZzWRuol0rcBF5TpJpbMquk2Wkf6LLAd-0-d04ad037ab48853b3c22b876c967bd86)
执行结果:
一次梯度下降=75.0;
二次梯度下降=150.0;
三次梯度下降=225.0。
二次、三次梯度下降应该都是75,结果都累加。
(8)使用x.grad.zero_()梯度重置代码如下。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P47_3221.jpg?sign=1739274621-EEtjIUiTbTfbCkAnIvuBZ2YwaHHLSTKT-0-5217c54dc28cae431377ad20a1cd24ac)
(9)多个变量梯度下降。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P47_3228.jpg?sign=1739274621-BzS9HUQAFJRXYBDKnnpBnNjIsrFrIEWQ-0-8e64ab1e632670ff34eb98db820735bf)
执行结果:z=6*(x^5)=18750。
接着改写【02_02_梯度下降法.ipynb】,将改用PyTorch函数微分。
(1)只更动微分函数:原来是自己手动计算,现改用自动微分。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P47_3236.jpg?sign=1739274621-EkgxS47rJQ8Gh775VyBXhZR7CzcGWjNV-0-e6979be46fcf8de7c4c1838c79f73470)
执行结果:与【02_02_梯度下降法.ipynb】相同。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P48_3256.jpg?sign=1739274621-c4h9lQw4NteqPiAIv8vd4aGcmYu0KUx2-0-ec41d77fd83ccb72849eee97da722822)
(2)再将函数改为2x4-3x2+2x-20。要缩小学习率(0.001),免得错过最小值,同时增大执行周期数(15000)。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P48_3259.jpg?sign=1739274621-ovowzr4DrkPwwVhC01MCZcYKuGW2ucLP-0-4befc5140e2b48dc7fe736d94bc529af)
执行结果:与【02_02_梯度下降法.ipynb】相同。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P48_3266.jpg?sign=1739274621-mdAuFq29jCmA38csZuBenRns84SgxQ6S-0-84da1cc9c2657df49fd0dc2ceece8036)
最后来操作一个完整范例,使用梯度下降法对线性回归求解,方程式如下,求w、b的最佳解。
y=wx+b
下列程序代码请参考【03_03_简单线性回归.ipynb】。
(1)载入套件。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P48_3269.jpg?sign=1739274621-vVi0Cqamufb2FWLVTSt2ZdoNuCpaXJbs-0-2c32efa5df83d7d44bfe777b626fcd91)
(2)定义训练函数:
· 刚开始w、b初始值均可设为任意值,这里使用正态分布之随机数。
· 定义损失函数=MSE,公式见11行。
· 依照2-3-3节证明,权重更新公式如下:
新权重=原权重-学习率(learning_rate)×梯度(gradient)
· 权重更新必须“设定不参与梯度下降”才能运算,参见第14~18行。
· 每一训练周期的w、b、损失函数都存储至数组,以利后续观察,参见第22~24行。要取得w、b、损失函数的值,可以使用.item()转为常数,也可以使用.detach().numpy()转为NumPy数组,detach作用是将变量脱离梯度下降的控制。
· 记得梯度重置,包括w、b。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P49_3289.jpg?sign=1739274621-Q933J2cmlYrGGq43Cn9r89YushEP5gjo-0-017f867553722f6d86cbc6773ad44bc4)
(3)产生线性随机数据100笔,介于0~50。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P49_3296.jpg?sign=1739274621-NLaXX64yFyzXA0qMetfAuGS568YqLx0u-0-1563f1b34c83e843982ee91ab102155e)
(4)执行训练。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P49_3303.jpg?sign=1739274621-bpZw7C7mTiNxfZcadxzWE9vSArbZRHWK-0-861024aa3e5f5c916a94813a5b11d57a)
执行结果:w=0.942326545715332, b=1.1824959516525269。
(5)执行训练100000次。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P49_3310.jpg?sign=1739274621-CR91LYzCsWq3WeKezQpqllnA07bJp0Y8-0-67546b67028961db3643414ea3734711)
执行结果有差异:w=0.8514814972877502, b=4.500218868255615。
(6)以NumPy验证。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P49_3317.jpg?sign=1739274621-dLToDUR5G49KWbOru0TRSStcVMrKFKiv-0-6e3368b63a9d9b2c766605e7284530ad)
执行结果:w=0.8510051491073364, b=4.517198474698629。结果与梯度下降法训练100000次较相近,显示梯度下降法收敛较慢,需要较多执行周期的训练,这与默认的学习率(lr)有关,读者可以调整反复测试。所以说深度学习必须靠试验与经验,才能找到最佳参数值。
(7)训练100次的模型绘图验证。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3339.jpg?sign=1739274621-mraSPjaP19YHNyHKf8DP19Ri7QNEOlt6-0-fc9cebeebc0aaefc22d765a3df345acd)
执行结果:虽然训练次数不足,但回归线也确实在样本点的中线。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3346.jpg?sign=1739274621-9jIaFTfo5S3U48tQXbwFC2fgtk5vwWbz-0-07a8b917ea8e46602e65b7f313a1b78d)
(8)NumPy模型绘图验证。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3349.jpg?sign=1739274621-LFyWfXpke9PZvEkQ6x9U4c87vZl4rfCs-0-d33177c218844540410407efa61e42b2)
执行结果:回归线在样本点的中线。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3356.jpg?sign=1739274621-2tR6Rc0VXkQfAPEmfW3FLwvUsSYQfBum-0-55d00ae15f2b5be25fd01ad8668b5f94)
(9)损失函数绘图验证。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3359.jpg?sign=1739274621-Euz04G6xOqx4D0H25GtQqxhAHORb7Mqo-0-ef9e3405ad51a481868944fce903a961)
执行结果:大约在第10个执行周期后就收敛了。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3366.jpg?sign=1739274621-JbzKnqVwZVSJIrzPPwsSlMRWBLDaxGeE-0-09b98999ba48e6894ecd510a550155fb)
有了PyTorch自动微分的功能,反向传导变得非常简单,若要改用其他损失函数,只须修改一下公式,其他程序代码都照旧就可以了。这一节模型训练的程序架构非常重要,只要熟悉每个环节,后续复杂的模型也可以运用自如。