PyTorch自动求导¶
作者:凯鲁嘎吉 - 博客园 http://www.cnblogs.com/kailugaji/
所用版本:python 3.6.5,torch 1.6.0,torchvision 0.7.0
假设我们想对函数$y=2\mathbf{x}^{\top}\mathbf{x}$关于列向量$\mathbf{x}$求导。
首先,我们创建变量x
并为其分配一个初始值。
import torch
x = torch.arange(4.0)
x
在计算$y$关于$\mathbf{x}$的梯度之前,需要一个地方来存储梯度。
重要的是,我们不会在每次对一个参数求导时都分配新的内存。因为我们经常会成千上万次地更新相同的参数,每次都分配新的内存可能很快就会将内存耗尽。注意,标量函数关于向量$\mathbf{x}$的梯度是向量,并且与$\mathbf{x}$具有相同的形状。
x.requires_grad_(True) # 等价于 `x = torch.arange(4.0, requires_grad=True)`
x.grad # 默认值是None
现在计算$y$
y = 2 * torch.dot(x, x)
y
x
是一个长度为4的向量,计算x
和x
的内积,得到了我们赋值给y
的标量输出。接下来,可以通过调用反向传播函数来自动计算y
关于x
每个分量的梯度,并打印这些梯度。
y.backward()
x.grad
函数$y=2\mathbf{x}^{\top}\mathbf{x}$关于$\mathbf{x}$的梯度应为$4\mathbf{x}$。让我们快速验证我们想要的梯度是否正确计算。
x.grad == 4 * x
现在让我们计算x
的另一个函数
# 在默认情况下,PyTorch会累积梯度,我们需要清除之前的值
x.grad.zero_()
# y=sum(x)
y = x.sum()
y.backward()
x.grad
# y=x的导数为1,因此计算出来是常数1
非标量变量的反向传播
# 对非标量调用`backward`需要传入一个`gradient`参数,该参数指定微分函数关于`self`的梯度。
# 在我们的例子中,我们只想求偏导数的和,所以传递一个1的梯度是合适的
x.grad.zero_()
y = x * x
# 等价于y.backward(torch.ones(len(x)))
y.sum().backward()
x.grad
# y=x^2的导数是2x
x.grad == 2 * x
分离计算
$z=x^3$,看成两部分,首先$y=x^2$,然后$z=yx$
x.grad.zero_()
y = x * x
在这里,我们可以分离y
来返回一个新变量u
,该变量与y
具有相同的值,但丢弃计算图中如何计算y
的任何信息。
换句话说,梯度不会向后流经u
到x
。因此,下面的反向传播函数计算z=u*x
关于x
的偏导数,同时将u
作为常数处理,而不是z=x*x*x
关于x
的偏导数。
u = y.detach()
z = u * x
z.sum().backward()
x.grad == u
由于记录了y
的计算结果,我们可以随后在y
上调用反向传播,得到y=x*x
关于的x
的导数,这里是2*x
。
x.grad.zero_()
y.sum().backward()
x.grad == 2 * x
Python控制流的梯度计算
def f(a):
b = a * 2
while b.norm() < 1000:
b = b * 2
if b.sum() > 0:
c = b
else:
c = 100 * b
return c
a = torch.randn(size=(), requires_grad=True)
d = f(a)
d.backward()
我们现在可以分析上面定义的f
函数。请注意,它在其输入a
中是分段线性的。换言之,对于任何a
,存在某个常量标量k
,使得f(a)=k*a
,其中k
的值取决于输入a
。因此,d/a
允许我们验证梯度是否正确。
a.grad == d / a
例子: 使$f(x)=\sin(x)$,绘制$f(x)$和$\frac{df(x)}{dx}$的图像,其中后者不使用$f'(x)=\cos(x)$。
import matplotlib.pyplot as plt
import torch
x = torch.arange(0.0,10,0.1)
x.requires_grad_(True)
x1 = x.detach()
y1 = torch.sin(x1)
y2 = torch.sin(x)
y2.sum().backward()
plt.plot(x1, y1, label='y=sin(x)')
plt.plot(x1, x.grad, label='dy/dx')
plt.xlabel('x')
plt.ylabel('y')
plt.legend()
plt.show()