![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=1739467761-fEBaeDV1cdr9wNLFP601Ns4tM7YgQc9z-0-8c6c6015a62b07b9144151502415b6e6)
图3.9 神经网络权重求解过程
下列程序代码请参考【03_02_自动微分.ipynb】。
(1)变量y会对x自动微分,代码如下。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P45_3069.jpg?sign=1739467761-fJ6DSDZaZMDIR6WppybmOQnIbswEoWXj-0-1b3206e9a3be771ac12f14970e3b30a0)
变量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=1739467761-SESjH044SrU4YuNXyAKnPwsCRSk7C5kB-0-3d4e0a08bb89b8ccd20cd5f6da3463bf)
(2)下列程序代码可取得自动微分相关的属性值。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P45_3086.jpg?sign=1739467761-xXgYJH1OIN7xiAPyvwH1k5DjSw3x4JZl-0-6feae09337db2b62b76d2016bf126732)
执行结果。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P45_3093.jpg?sign=1739467761-1P1OgAeje91GYVTHX8QHo63M7jII7v3G-0-e03b32dd44edd100f8fb556df90ef2ed)
(3)来看一个较复杂的例子,以神经网络进行分类时,常使用交叉熵(Cross Entropy)作为损失函数,下图表达Cross Entropy=CE(y, wx+b)。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_25835.jpg?sign=1739467761-xHk3W3pFQvgvcunoClgBt4lxphSwRErv-0-78992e6300fcfbddf29df630a64b4d1b)
图3.10 交叉熵(Cross Entropy)运算图
PyTorch会依据程序,建构运算图(Computational Graph),描述梯度下降时,变量运算的顺序如下所示,先算x,再算z,最后计算loss。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_3165.jpg?sign=1739467761-M3qHPFtWP768vul9sdZ14ZGKk5aiThN5-0-51991c3698da183ea15cb3fc3138baaf)
torch.nn.functional.binary_cross_entropy_with_logits是PyTorch提供的交叉熵函数。
执行结果:
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_3172.jpg?sign=1739467761-hqTYSpebU6OIbizz2esXjSupehIm0Y6s-0-ef199886642ff102b1167f3482a5b63e)
(4)自动微分中,z是w、b的函数,而loss又是z的函数,故只要对loss进行反向传导即可。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_3175.jpg?sign=1739467761-9o31EcrInJOSzTp2Lg6qCh8ak1hZ9M7D-0-2e964c97697ec6bbf9e6dfa581e71517)
执行结果中,w、b梯度分别为:
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_3182.jpg?sign=1739467761-lmKObMJJZZZCAkUVLyZZFuf53OMyy0XL-0-8cfcc7e578bcbcdb960813fb1b1fc5ea)
(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=1739467761-OY1jTcXfTvV8ut9MV0hPWa393vp3F0iy-0-0ee51461f863368f816acb4392a4d58a)
上例以torch.ones替代Variable。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P46_3192.jpg?sign=1739467761-FU6gaNU9lONE5qINQeKtf2EM2PZthmWt-0-ed9067d8234d3b8cbf444fe42ef20a1b)
(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=1739467761-qRoVekmIOCygvvioHHKSlSp6fJSYwdHF-0-c7ab3cbc6925c70fdd3ba92984daee66)
执行结果:
一次梯度下降=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=1739467761-iuVpCttYDwQVZmm9WqXu1pZJ60NEvjR1-0-54f6bef892f184a9d87195dfbd91c6b2)
(9)多个变量梯度下降。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P47_3228.jpg?sign=1739467761-ZrtTOCyoBcb4sT0PRjgHiZh69BrFbA72-0-43c9a3cd2dc9798a2e5c8c8788a8cc7a)
执行结果:z=6*(x^5)=18750。
接着改写【02_02_梯度下降法.ipynb】,将改用PyTorch函数微分。
(1)只更动微分函数:原来是自己手动计算,现改用自动微分。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P47_3236.jpg?sign=1739467761-i6c29745unRGERmCEkIehzLeC67FppBE-0-cedd745dd5bb72425438bf5cfb492c1b)
执行结果:与【02_02_梯度下降法.ipynb】相同。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P48_3256.jpg?sign=1739467761-plP9lajVAy5uqCQhok5MY9YMYbbrcp7m-0-4d4fdbc73e79f9131a68c775df8ed20d)
(2)再将函数改为2x4-3x2+2x-20。要缩小学习率(0.001),免得错过最小值,同时增大执行周期数(15000)。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P48_3259.jpg?sign=1739467761-hHMsb38npZYRGdxVv8v0gqGXmDCbNTtp-0-3e1b7fc4ba7298aeed0b7a8691df13a7)
执行结果:与【02_02_梯度下降法.ipynb】相同。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P48_3266.jpg?sign=1739467761-dPWNvOGGvgUUp2cDR3BArinetTSPUnPX-0-7e1916f2d120e3bb25c05b9828eaf67b)
最后来操作一个完整范例,使用梯度下降法对线性回归求解,方程式如下,求w、b的最佳解。
y=wx+b
下列程序代码请参考【03_03_简单线性回归.ipynb】。
(1)载入套件。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P48_3269.jpg?sign=1739467761-rWl7hNL2kbtGZtxGkViNohVLwT2rlxH7-0-fddc1fe649309b0b17640ecd3150e1cb)
(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=1739467761-6w5kAEBn3yEqF19JSkLfqevUZXWgvZBS-0-05109a957ce4c907508ce48881013eed)
(3)产生线性随机数据100笔,介于0~50。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P49_3296.jpg?sign=1739467761-pKXpdkXdbcPnkNtWp8ppbynhjV5Kubdz-0-1cbf6bcee8268b8f21594fda5681a847)
(4)执行训练。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P49_3303.jpg?sign=1739467761-EK9fSWuoRndin7crzJ01PFaFREPsMl5f-0-9caddd7c9f763d10a2c47a8d5321b8a0)
执行结果:w=0.942326545715332, b=1.1824959516525269。
(5)执行训练100000次。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P49_3310.jpg?sign=1739467761-URa5m9U5wEZHgEPKclAAsfvXjmaVtxll-0-77faf7b0a692c35b2fb17c5733acfa7b)
执行结果有差异:w=0.8514814972877502, b=4.500218868255615。
(6)以NumPy验证。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P49_3317.jpg?sign=1739467761-NurdBEi8SX28ApRF8EXUBlSnxzWvjKlw-0-80c6cbef9dd58e2f08d37c25104a5f4b)
执行结果:w=0.8510051491073364, b=4.517198474698629。结果与梯度下降法训练100000次较相近,显示梯度下降法收敛较慢,需要较多执行周期的训练,这与默认的学习率(lr)有关,读者可以调整反复测试。所以说深度学习必须靠试验与经验,才能找到最佳参数值。
(7)训练100次的模型绘图验证。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3339.jpg?sign=1739467761-7aafHBrGs745SvwEVx54Da8OjLEdy5iq-0-25c056fde4a5af6d002d29a4c9f4d779)
执行结果:虽然训练次数不足,但回归线也确实在样本点的中线。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3346.jpg?sign=1739467761-SvmRuurAj3aknK8eQnn1tEyFK1ic7TZr-0-cb2f9fa5350267180566daeca6b0617a)
(8)NumPy模型绘图验证。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3349.jpg?sign=1739467761-vHOeIwl6p7tKFGMBEyKRpczlvclMnDDb-0-85d4cc646300c38deb37eb523564ab61)
执行结果:回归线在样本点的中线。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3356.jpg?sign=1739467761-iv8ExW3xrTBtoPRqpvpLiypasNBeVnMq-0-ed857dcb92d182a4b68234173597fa41)
(9)损失函数绘图验证。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3359.jpg?sign=1739467761-NJnW7wzZC8lgCwhquCzHkI0FqR6rWh0J-0-0edb012c65e901e6a45b7cfbc143cf05)
执行结果:大约在第10个执行周期后就收敛了。
![](https://epubservercos.yuewen.com/128DEE/31397898903670606/epubprivate/OEBPS/Images/Figure-P50_3366.jpg?sign=1739467761-oVirJMP05NU8tsZADPo0YuYXxgEDx1ca-0-c6eaeb7f725a0e18cb168536977967dc)
有了PyTorch自动微分的功能,反向传导变得非常简单,若要改用其他损失函数,只须修改一下公式,其他程序代码都照旧就可以了。这一节模型训练的程序架构非常重要,只要熟悉每个环节,后续复杂的模型也可以运用自如。