在深度学习中,我们经常需要对函数求梯度。PyTorch提供的autograd包能够根据输入和前向传播过程自动构建计算图,并执行反向传播。本节将介绍如何使用autograd包来进行自动求梯度的有关操作


引言

Tensor是这个包的核心类,如果将其属性.requires_grad设置为True,它将开始追踪(track)在其上的所有操作(这样就可以利用链式法则进行梯度传播了)。完成计算后,可以调用.backward()来完成所有梯度计算。此Tensor的梯度将累积到.grad属性中

注意在y.backward()时,如果y是标量,则不需要为backward()传入任何参数;否则,需要传入一个与y同形的Tensor

如果不想要被继续追踪,可以调用.detach()将其从追踪记录中分离出来,这样就可以防止将来的计算被追踪,这样梯度就传不过去了。此外,还可以用with torch.no_grad()将不想被追踪的操作代码块包裹起来,这种方法在评估模型的时候很常用,因为在评估模型时,我们并不需要计算可训练参数(requires_grad = True)的梯度
Function是另外一个很重要的类。Tensor和Function互相结合就可以构建一个记录有整个计算过程的有向无环图(DAG)。每个Tensor都有一个.grad_fn属性,该属性即创建该Tensor的Function, 就是说该Tensor是不是通过某些运算得到的,若是,则grad_fn返回一个与这些运算相关的对象,否则是None


Tensor<br />

创建一个Tensor并设置requires_grad = True:

  1. x = torch.ones(2, 2, requires_grad=True)
  2. print(x)
  3. print(x.grad_fn)
  4. tensor([[1., 1.],
  5. [1., 1.]], requires_grad=True)
  6. None

运算操作:

  1. y = x + 2
  2. print(y)
  3. print(y.grad_fn)
  4. tensor([[3., 3.],
  5. [3., 3.]], grad_fn=<AddBackward>)
  6. <AddBackward object at 0x1100477b8>

x是直接创建的,所以它没有grad_fn, 而y是通过一个加法操作创建的,所以它有一个为的grad_fn。像x这种直接创建的称为叶子节点,叶子节点对应的grad_fn是None。

  1. print(x.is_leaf, y.is_leaf)
  2. True
  3. False

复杂度运算:

  1. z = y * y * 3
  2. out = z.mean()
  3. print(z, out)
  4. tensor([[27., 27.],
  5. [27., 27.]], grad_fn=<MulBackward>) tensor(27., grad_fn=<MeanBackward1>)

通过.requiresgrad()来用inplace的方式改变requires_grad属性:

  1. a = torch.randn(2, 2) # 缺失情况下默认 requires_grad = False
  2. a = ((a * 3) / (a - 1))
  3. print(a.requires_grad) # False
  4. a.requires_grad_(True)
  5. print(a.requires_grad) # True
  6. b = (a * a).sum()
  7. print(b.grad_fn)
  8. False
  9. True
  10. <SumBackward0 object at 0x118f50cc0>

梯度

因为out是一个标量,所以调用backward()时不需要指定求导变量:

  1. out.backward() # 等价于 out.backward(torch.tensor(1.))

我们来看看out关于x的梯度自动求梯度 - 图1:

  1. print(x.grad)
  2. tensor([[4.5000, 4.5000],
  3. [4.5000, 4.5000]])

我们令out的时间复杂度为o , 因为
自动求梯度 - 图2

所以
自动求梯度 - 图3

所以上面的输出是正确的

数学上,如果有一个函数值和自变量都为向量的函数自动求梯度 - 图4, 那么 自动求梯度 - 图5 关于 自动求梯度 - 图6 的梯度就是一个雅可比矩阵:

自动求梯度 - 图7

而torch.autograd这个包就是用来计算一些雅克比矩阵的乘积的
例如,如果 自动求梯度 - 图8 是一个标量函数的 自动求梯度 - 图9 的梯度:

自动求梯度 - 图10

那么根据链式法则,l 关于 自动求梯度 - 图11 的雅克比矩阵就为:

自动求梯度 - 图12

注意:grad在反向传播过程中是累加的,这意味着每一次运行反向传播,梯度都会累加之前的梯度,所以一般在反向传播之前需把梯度清零

  1. # 再来反向传播一次,注意grad是累加的
  2. out2 = x.sum()
  3. out2.backward()
  4. print(x.grad)
  5. out3 = x.sum()
  6. x.grad.data.zero_()
  7. out3.backward()
  8. print(x.grad)
  9. tensor([[5.5000, 5.5000],
  10. [5.5000, 5.5000]])
  11. tensor([[1., 1.],
  12. [1., 1.]])

为什么在y.backward()时,如果y是标量,则不需要为backward()传入任何参数;否则,需要传入一个与y同形的Tensor?
简单来说就是为了避免向量(甚至更高维张量)对张量求导,而转换成标量对张量求导。举个例子,假设形状为 自动求梯度 - 图13 的矩阵 X 经过运算得到了 自动求梯度 - 图14 的矩阵 Y,Y 又经过运算得到了 自动求梯度 - 图15 的矩阵 Z。那么按照前面讲的规则,自动求梯度 - 图16 应该是一个自动求梯度 - 图17 四维张量,自动求梯度 - 图18 是一个 自动求梯度 - 图19的四维张量。问题来了,怎样反向传播?怎样将两个四维张量相乘??这要怎么乘???就算能解决两个四维张量怎么乘的问题,四维和三维的张量又怎么乘?导数的导数又怎么求?
为了避免这个问题,我们不允许张量对张量求导,只允许标量对张量求导,求导结果是和自变量同形的张量。所以必要时我们要把张量通过将所有张量的元素加权求和的方式转换为标量,举个例子,假设y由自变量x计算而来,w是和y同形的张量,则y.backward(w)的含义是:先计算l = torch.sum(y * w),则l是个标量,然后求l对自变量x的导数

  1. x = torch.tensor([1.0, 2.0, 3.0, 4.0], requires_grad=True)
  2. y = 2 * x
  3. z = y.view(2, 2)
  4. print(z)
  5. tensor([[2., 4.],
  6. [6., 8.]], grad_fn=<ViewBackward>)

现在 z 不是一个标量,所以在调用backward时需要传入一个和z同形的权重向量进行加权求和得到一个标量

  1. v = torch.tensor([[1.0, 0.1], [0.01, 0.001]], dtype=torch.float)
  2. z.backward(v)
  3. print(x.grad)
  4. tensor([2.0000, 0.2000, 0.0200, 0.0020])

注意,x.grad是和x同形的张量

再来看看中断梯度追踪的例子:

  1. x = torch.tensor(1.0, requires_grad=True)
  2. y1 = x ** 2
  3. with torch.no_grad():
  4. y2 = x ** 3 #指数运算
  5. y3 = y1 + y2
  6. print(x.requires_grad)
  7. print(y1, y1.requires_grad) # True
  8. print(y2, y2.requires_grad) # False
  9. print(y3, y3.requires_grad) # True
  10. True
  11. tensor(1., grad_fn=<PowBackward0>) True
  12. tensor(1.) False
  13. tensor(2., grad_fn=<ThAddBackward>) True

可以看到,上面的y2是没有grad_fn而且y2.requires_grad=False的,而y3是有grad_fn的。如果我们将y3对x求梯度的话会是多少呢?

  1. y3.backward()
  2. print(x.grad)
  3. tensor(2.)

为什么是2呢?
自动求梯度 - 图20,当自动求梯度 - 图21自动求梯度 - 图22不应该是5吗?事实上,由于 自动求梯度 - 图23 的定义是被torch.no_grad():包裹的,所以与 自动求梯度 - 图24有关的梯度是不会回传的,只有与自动求梯度 - 图25 有关的梯度才会回传,即 自动求梯度 - 图26自动求梯度 - 图27 的梯度
上面提到,y2.requires_grad=False,所以不能调用 y2.backward(),会报错:

  1. RuntimeError: element 0 of tensors does not require grad and does not have a grad_fn

此外,如果我们想要修改tensor的数值,但是又不希望被autograd记录(即不会影响反向传播),那么我么可以对tensor.data进行操作

  1. x = torch.ones(1,requires_grad=True)
  2. print(x.data) # 还是一个tensor
  3. print(x.data.requires_grad) # 但是已经是独立于计算图之外
  4. y = 2 * x
  5. x.data *= 100 # 只改变了值,不会记录在计算图,所以不会影响梯度传播
  6. y.backward()
  7. print(x) # 更改data的值也会影响tensor的值
  8. print(x.grad)
  9. tensor([1.])
  10. False
  11. tensor([100.], requires_grad=True)
  12. tensor([2.])