![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=1738748182-tv4LXtLceNg5q7ndMC7ACbgYymTsGDSW-0-ab1daa1b8be77704324ac8446cdfa85a)
图3.9 神经网络权重求解过程
下列程序代码请参考【03_02_自动微分.ipynb】。
(1)变量y会对x自动微分,代码如下。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P45_3069.jpg?sign=1738748182-folA9hRiLPFkG7nV4ZpZpp2WYDXdwxgu-0-a212917ab4f90c6034f70e86327d1e7f)
变量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=1738748182-XMD53heHHRE8R0TZ13Ag8QO5BxDaYSBU-0-4ba4501cd56babe6ead92fd995091236)
(2)下列程序代码可取得自动微分相关的属性值。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P45_3086.jpg?sign=1738748182-W1W3H5YRFd56Ecju6WWXsjpWuiPe1umq-0-e315f354b1e00d41f389fb3e2dd08a26)
执行结果。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P45_3093.jpg?sign=1738748182-4ftb7ju3Vv3QfDEeI65pfR9LEjKhy2wP-0-2c27c231957db411bce8a210d2f91034)
(3)来看一个较复杂的例子,以神经网络进行分类时,常使用交叉熵(Cross Entropy)作为损失函数,下图表达Cross Entropy=CE(y, wx+b)。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_25835.jpg?sign=1738748182-DcdFexjKKt7rDyBt1YjaMHNtSJRItaXH-0-955b7d6579c6485523ec997ab517858d)
图3.10 交叉熵(Cross Entropy)运算图
PyTorch会依据程序,建构运算图(Computational Graph),描述梯度下降时,变量运算的顺序如下所示,先算x,再算z,最后计算loss。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_3165.jpg?sign=1738748182-9mL3Zv1wHgx0RRNPvYxbpTpYZYRq278J-0-d7845227f783f411b4c26427196b6e83)
torch.nn.functional.binary_cross_entropy_with_logits是PyTorch提供的交叉熵函数。
执行结果:
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_3172.jpg?sign=1738748182-5DDX3qEpEeq3U51yobzFcTOe1U71jDYr-0-1d117d9f72a384eacfa0058e7b88c160)
(4)自动微分中,z是w、b的函数,而loss又是z的函数,故只要对loss进行反向传导即可。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_3175.jpg?sign=1738748182-dHsObh9KrR68mzeE2M2lsQdXXewDdE1T-0-2cf698205a8f0229ae971db315a8a48c)
执行结果中,w、b梯度分别为:
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_3182.jpg?sign=1738748182-KQ4myVQfoMJcDc5NDFN9wz5NYuJojcst-0-7bfc0295e2e9f49598f0819376747a00)
(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=1738748182-YSLsLX5ckJBAyG4dOkpxzwUysG87rDZ2-0-cb44305dba846266360d434907ee51f8)
上例以torch.ones替代Variable。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_3192.jpg?sign=1738748182-FhvN2JNWp9NK28i7FJVVb5aTGCKJEWVV-0-fafb9811c9d4d77268bd9e48ee768394)
(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=1738748182-WlStCSD6Q4k6CIIi0eoK4gtrwEepjoGR-0-577ca43bc24ed6f06feeaa9f7338cbcb)
执行结果:
一次梯度下降=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=1738748182-P7ASC7c7SyuSXBenNBmrFsioZgMND1fR-0-964448b0126014c4e6016b46cda9b7cb)
(9)多个变量梯度下降。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P47_3228.jpg?sign=1738748182-nBMlYnqRuFuARoCOK0xKTQTeqxnEVekm-0-99d39016f5a7122e07a25115bb637a29)
执行结果:z=6*(x^5)=18750。
接着改写【02_02_梯度下降法.ipynb】,将改用PyTorch函数微分。
(1)只更动微分函数:原来是自己手动计算,现改用自动微分。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P47_3236.jpg?sign=1738748182-RYUvwE24gUW85Ix7PcrbDGphE8eAXhT0-0-3ac9040524c8ab2971bfe14cbeb0c126)
执行结果:与【02_02_梯度下降法.ipynb】相同。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P48_3256.jpg?sign=1738748182-OPu8XchsgBC6Q0o5Y163Cl1EtEwkmNqX-0-934916c6874da8db4a6b069aacac28b6)
(2)再将函数改为2x4-3x2+2x-20。要缩小学习率(0.001),免得错过最小值,同时增大执行周期数(15000)。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P48_3259.jpg?sign=1738748182-GzexVf6eVEvQE4GYkbR3FOXHIsbz7yJE-0-4055bf6d1148ff4dfe0b716b6e511e03)
执行结果:与【02_02_梯度下降法.ipynb】相同。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P48_3266.jpg?sign=1738748182-7akpBGvG6KE8HRrqX92mCTiQe9vZYOmd-0-cdcd9ceb2100b5c4741ee1e9c868c9b4)
最后来操作一个完整范例,使用梯度下降法对线性回归求解,方程式如下,求w、b的最佳解。
y=wx+b
下列程序代码请参考【03_03_简单线性回归.ipynb】。
(1)载入套件。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P48_3269.jpg?sign=1738748182-ljNsPPf7HvabosjHK2VNFTBFu6OJTcZK-0-db66a5ea2c8351f9fec217fc2ea1c8f2)
(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=1738748182-qwCE5jZvlwiB3FAV9godr5tCfFdkvypt-0-6c363d8bf74a2b681602e26bdf764a8b)
(3)产生线性随机数据100笔,介于0~50。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P49_3296.jpg?sign=1738748182-LGqjQcjJS2OqNVq4geCjtRszFXOABY9P-0-5c0fc37435294ff89466845402c45d21)
(4)执行训练。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P49_3303.jpg?sign=1738748182-RaWPh4MrujGu7vEPru8reyTHcWDrfXcM-0-562138871457af3bce02c8b7797ba91e)
执行结果:w=0.942326545715332, b=1.1824959516525269。
(5)执行训练100000次。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P49_3310.jpg?sign=1738748182-nBNW71lISS6KfYe2Dnt0Me9NTNvVQ1p9-0-4f472d5f75fa84e249e3d5108c75c08b)
执行结果有差异:w=0.8514814972877502, b=4.500218868255615。
(6)以NumPy验证。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P49_3317.jpg?sign=1738748182-5zxiK1jmYCFxl9nDmFdTUyhTF70DxXf2-0-66c23bea2be513d6e1b8275cdcf4cfdd)
执行结果:w=0.8510051491073364, b=4.517198474698629。结果与梯度下降法训练100000次较相近,显示梯度下降法收敛较慢,需要较多执行周期的训练,这与默认的学习率(lr)有关,读者可以调整反复测试。所以说深度学习必须靠试验与经验,才能找到最佳参数值。
(7)训练100次的模型绘图验证。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3339.jpg?sign=1738748182-wPc1ugvoJAO1f9rwAe0CCvtCudIIHn93-0-a6c9354c0e5546d0d465e405e3c89e95)
执行结果:虽然训练次数不足,但回归线也确实在样本点的中线。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3346.jpg?sign=1738748182-ybHRO44w6ZzkvABiolMOXjoe3SGSTJwO-0-07c821f12ee165cd91a57160ef28b90d)
(8)NumPy模型绘图验证。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3349.jpg?sign=1738748182-gV2GEF6jwOdgL5JOXZoxlP1EHy1sviSq-0-ffecbbc02e0d328065641904f47d64c7)
执行结果:回归线在样本点的中线。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3356.jpg?sign=1738748182-ZCbeYCi1b0OWy1unYXIgXm4wx9g9ZZfZ-0-b0f78d675a1273506105a3a565f6e8d4)
(9)损失函数绘图验证。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3359.jpg?sign=1738748182-353Y9dxnl0PrDGNi1764Zt81sOe2HnNC-0-c2e613a2ccd671509a2064a258e83a09)
执行结果:大约在第10个执行周期后就收敛了。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3366.jpg?sign=1738748182-T7PmodzV32dysUAjP9pf54T6GtvDsF5v-0-c184332e97d061babdcdf83d7ac4136f)
有了PyTorch自动微分的功能,反向传导变得非常简单,若要改用其他损失函数,只须修改一下公式,其他程序代码都照旧就可以了。这一节模型训练的程序架构非常重要,只要熟悉每个环节,后续复杂的模型也可以运用自如。