上一节中我们观察了过拟合现象,即模型的训练误差远小于它在测试集上的误差。虽然增大训练数据集可能会减轻过拟合,但是获取额外的训练数据往往代价高昂。本节介绍应对过拟合问题的常用方法:权重衰减(weight decay)。

3.12.1 方法

权重衰减等价于 3.12 权重衰减 - 图2 范数正则化(regularization)。正则化通过为模型损失函数添加惩罚项使学出的模型参数值较小,是应对过拟合的常用手段。我们先描述3.12 权重衰减 - 图3范数正则化,再解释它为何又称权重衰减。

3.12 权重衰减 - 图4范数正则化在模型原损失函数基础上添加3.12 权重衰减 - 图5范数惩罚项,从而得到训练所需要最小化的函数。3.12 权重衰减 - 图6范数惩罚项指的是模型权重参数每个元素的平方和与一个正的常数的乘积。以3.1节(线性回归)中的线性回归损失函数

3.12 权重衰减 - 图7%20%3D%20%5Cfrac%7B1%7D%7Bn%7D%20%5Csum%7Bi%3D1%7D%5En%20%5Cfrac%7B1%7D%7B2%7D%5Cleft(x_1%5E%7B(i)%7D%20w_1%20%2B%20x_2%5E%7B(i)%7D%20w_2%20%2B%20b%20-%20y%5E%7B(i)%7D%5Cright)%5E2%0A#card=math&code=%5Cell%28w_1%2C%20w_2%2C%20b%29%20%3D%20%5Cfrac%7B1%7D%7Bn%7D%20%5Csum%7Bi%3D1%7D%5En%20%5Cfrac%7B1%7D%7B2%7D%5Cleft%28x_1%5E%7B%28i%29%7D%20w_1%20%2B%20x_2%5E%7B%28i%29%7D%20w_2%20%2B%20b%20-%20y%5E%7B%28i%29%7D%5Cright%29%5E2%0A)

为例,其中3.12 权重衰减 - 图8是权重参数,3.12 权重衰减 - 图9是偏差参数,样本3.12 权重衰减 - 图10的输入为3.12 权重衰减 - 图11%7D%2C%20x_2%5E%7B(i)%7D#card=math&code=x_1%5E%7B%28i%29%7D%2C%20x_2%5E%7B%28i%29%7D),标签为3.12 权重衰减 - 图12%7D#card=math&code=y%5E%7B%28i%29%7D),样本数为3.12 权重衰减 - 图13。将权重参数用向量3.12 权重衰减 - 图14表示,带有3.12 权重衰减 - 图15范数惩罚项的新损失函数为

3.12 权重衰减 - 图16%20%2B%20%5Cfrac%7B%5Clambda%7D%7B2n%7D%20%5C%7C%5Cboldsymbol%7Bw%7D%5C%7C%5E2%2C%0A#card=math&code=%5Cell%28w_1%2C%20w_2%2C%20b%29%20%2B%20%5Cfrac%7B%5Clambda%7D%7B2n%7D%20%5C%7C%5Cboldsymbol%7Bw%7D%5C%7C%5E2%2C%0A)

其中超参数3.12 权重衰减 - 图17。当权重参数均为0时,惩罚项最小。当3.12 权重衰减 - 图18较大时,惩罚项在损失函数中的比重较大,这通常会使学到的权重参数的元素较接近0。当3.12 权重衰减 - 图19设为0时,惩罚项完全不起作用。上式中3.12 权重衰减 - 图20范数平方3.12 权重衰减 - 图21展开后得到3.12 权重衰减 - 图22。有了3.12 权重衰减 - 图23范数惩罚项后,在小批量随机梯度下降中,我们将线性回归一节中权重3.12 权重衰减 - 图243.12 权重衰减 - 图25的迭代方式更改为

3.12 权重衰减 - 图26w1%20-%20%20%20%5Cfrac%7B%5Ceta%7D%7B%7C%5Cmathcal%7BB%7D%7C%7D%20%5Csum%7Bi%20%5Cin%20%5Cmathcal%7BB%7D%7Dx1%5E%7B(i)%7D%20%5Cleft(x_1%5E%7B(i)%7D%20w_1%20%2B%20x_2%5E%7B(i)%7D%20w_2%20%2B%20b%20-%20y%5E%7B(i)%7D%5Cright)%2C%5C%5C%0Aw_2%20%26%5Cleftarrow%20%5Cleft(1-%20%5Cfrac%7B%5Ceta%5Clambda%7D%7B%7C%5Cmathcal%7BB%7D%7C%7D%20%5Cright)w_2%20-%20%20%20%5Cfrac%7B%5Ceta%7D%7B%7C%5Cmathcal%7BB%7D%7C%7D%20%5Csum%7Bi%20%5Cin%20%5Cmathcal%7BB%7D%7Dx2%5E%7B(i)%7D%20%5Cleft(x_1%5E%7B(i)%7D%20w_1%20%2B%20x_2%5E%7B(i)%7D%20w_2%20%2B%20b%20-%20y%5E%7B(i)%7D%5Cright).%0A%5Cend%7Baligned%7D%0A#card=math&code=%5Cbegin%7Baligned%7D%0Aw_1%20%26%5Cleftarrow%20%5Cleft%281-%20%5Cfrac%7B%5Ceta%5Clambda%7D%7B%7C%5Cmathcal%7BB%7D%7C%7D%20%5Cright%29w_1%20-%20%20%20%5Cfrac%7B%5Ceta%7D%7B%7C%5Cmathcal%7BB%7D%7C%7D%20%5Csum%7Bi%20%5Cin%20%5Cmathcal%7BB%7D%7Dx1%5E%7B%28i%29%7D%20%5Cleft%28x_1%5E%7B%28i%29%7D%20w_1%20%2B%20x_2%5E%7B%28i%29%7D%20w_2%20%2B%20b%20-%20y%5E%7B%28i%29%7D%5Cright%29%2C%5C%5C%0Aw_2%20%26%5Cleftarrow%20%5Cleft%281-%20%5Cfrac%7B%5Ceta%5Clambda%7D%7B%7C%5Cmathcal%7BB%7D%7C%7D%20%5Cright%29w_2%20-%20%20%20%5Cfrac%7B%5Ceta%7D%7B%7C%5Cmathcal%7BB%7D%7C%7D%20%5Csum%7Bi%20%5Cin%20%5Cmathcal%7BB%7D%7Dx_2%5E%7B%28i%29%7D%20%5Cleft%28x_1%5E%7B%28i%29%7D%20w_1%20%2B%20x_2%5E%7B%28i%29%7D%20w_2%20%2B%20b%20-%20y%5E%7B%28i%29%7D%5Cright%29.%0A%5Cend%7Baligned%7D%0A)

可见,3.12 权重衰减 - 图27范数正则化令权重3.12 权重衰减 - 图283.12 权重衰减 - 图29先自乘小于1的数,再减去不含惩罚项的梯度。因此,3.12 权重衰减 - 图30范数正则化又叫权重衰减。权重衰减通过惩罚绝对值较大的模型参数为需要学习的模型增加了限制,这可能对过拟合有效。实际场景中,我们有时也在惩罚项中添加偏差元素的平方和。

3.12.2 高维线性回归实验

下面,我们以高维线性回归为例来引入一个过拟合问题,并使用权重衰减来应对过拟合。设数据样本特征的维度为3.12 权重衰减 - 图31。对于训练数据集和测试数据集中特征为3.12 权重衰减 - 图32的任一样本,我们使用如下的线性函数来生成该样本的标签:

3.12 权重衰减 - 图33

其中噪声项3.12 权重衰减 - 图34服从均值为0、标准差为0.01的正态分布。为了较容易地观察过拟合,我们考虑高维线性回归问题,如设维度3.12 权重衰减 - 图35;同时,我们特意把训练数据集的样本数设低,如20。

  1. %matplotlib inline
  2. import torch
  3. import torch.nn as nn
  4. import numpy as np
  5. import sys
  6. sys.path.append("..")
  7. import d2lzh_pytorch as d2l
  8. n_train, n_test, num_inputs = 20, 100, 200
  9. true_w, true_b = torch.ones(num_inputs, 1) * 0.01, 0.05
  10. features = torch.randn((n_train + n_test, num_inputs))
  11. labels = torch.matmul(features, true_w) + true_b
  12. labels += torch.tensor(np.random.normal(0, 0.01, size=labels.size()), dtype=torch.float)
  13. train_features, test_features = features[:n_train, :], features[n_train:, :]
  14. train_labels, test_labels = labels[:n_train], labels[n_train:]

3.12.3 从零开始实现

下面先介绍从零开始实现权重衰减的方法。我们通过在目标函数后添加3.12 权重衰减 - 图36范数惩罚项来实现权重衰减。

3.12.3.1 初始化模型参数

首先,定义随机初始化模型参数的函数。该函数为每个参数都附上梯度。

  1. def init_params():
  2. w = torch.randn((num_inputs, 1), requires_grad=True)
  3. b = torch.zeros(1, requires_grad=True)
  4. return [w, b]

3.12.3.2 定义3.12 权重衰减 - 图37范数惩罚项

下面定义3.12 权重衰减 - 图38范数惩罚项。这里只惩罚模型的权重参数。

  1. def l2_penalty(w):
  2. return (w**2).sum() / 2

3.12.3.3 定义训练和测试

下面定义如何在训练数据集和测试数据集上分别训练和测试模型。与前面几节中不同的是,这里在计算最终的损失函数时添加了3.12 权重衰减 - 图39范数惩罚项。

  1. batch_size, num_epochs, lr = 1, 100, 0.003
  2. net, loss = d2l.linreg, d2l.squared_loss
  3. dataset = torch.utils.data.TensorDataset(train_features, train_labels)
  4. train_iter = torch.utils.data.DataLoader(dataset, batch_size, shuffle=True)
  5. def fit_and_plot(lambd):
  6. w, b = init_params()
  7. train_ls, test_ls = [], []
  8. for _ in range(num_epochs):
  9. for X, y in train_iter:
  10. # 添加了L2范数惩罚项
  11. l = loss(net(X, w, b), y) + lambd * l2_penalty(w)
  12. l = l.sum()
  13. if w.grad is not None:
  14. w.grad.data.zero_()
  15. b.grad.data.zero_()
  16. l.backward()
  17. d2l.sgd([w, b], lr, batch_size)
  18. train_ls.append(loss(net(train_features, w, b), train_labels).mean().item())
  19. test_ls.append(loss(net(test_features, w, b), test_labels).mean().item())
  20. d2l.semilogy(range(1, num_epochs + 1), train_ls, 'epochs', 'loss',
  21. range(1, num_epochs + 1), test_ls, ['train', 'test'])
  22. print('L2 norm of w:', w.norm().item())

3.12.3.4 观察过拟合

接下来,让我们训练并测试高维线性回归模型。当lambd设为0时,我们没有使用权重衰减。结果训练误差远小于测试集上的误差。这是典型的过拟合现象。

  1. fit_and_plot(lambd=0)

输出:

  1. L2 norm of w: 15.114808082580566

3.12_output1.png

3.12.3.5 使用权重衰减

下面我们使用权重衰减。可以看出,训练误差虽然有所提高,但测试集上的误差有所下降。过拟合现象得到一定程度的缓解。另外,权重参数的3.12 权重衰减 - 图41范数比不使用权重衰减时的更小,此时的权重参数更接近0。

  1. fit_and_plot(lambd=3)

输出:

  1. L2 norm of w: 0.035220853984355927

3.12_output2.png

3.12.4 简洁实现

这里我们直接在构造优化器实例时通过weight_decay参数来指定权重衰减超参数。默认下,PyTorch会对权重和偏差同时衰减。我们可以分别对权重和偏差构造优化器实例,从而只对权重衰减。

  1. def fit_and_plot_pytorch(wd):
  2. # 对权重参数衰减。权重名称一般是以weight结尾
  3. net = nn.Linear(num_inputs, 1)
  4. nn.init.normal_(net.weight, mean=0, std=1)
  5. nn.init.normal_(net.bias, mean=0, std=1)
  6. optimizer_w = torch.optim.SGD(params=[net.weight], lr=lr, weight_decay=wd) # 对权重参数衰减
  7. optimizer_b = torch.optim.SGD(params=[net.bias], lr=lr) # 不对偏差参数衰减
  8. train_ls, test_ls = [], []
  9. for _ in range(num_epochs):
  10. for X, y in train_iter:
  11. l = loss(net(X), y).mean()
  12. optimizer_w.zero_grad()
  13. optimizer_b.zero_grad()
  14. l.backward()
  15. # 对两个optimizer实例分别调用step函数,从而分别更新权重和偏差
  16. optimizer_w.step()
  17. optimizer_b.step()
  18. train_ls.append(loss(net(train_features), train_labels).mean().item())
  19. test_ls.append(loss(net(test_features), test_labels).mean().item())
  20. d2l.semilogy(range(1, num_epochs + 1), train_ls, 'epochs', 'loss',
  21. range(1, num_epochs + 1), test_ls, ['train', 'test'])
  22. print('L2 norm of w:', net.weight.data.norm().item())

与从零开始实现权重衰减的实验现象类似,使用权重衰减可以在一定程度上缓解过拟合问题。

  1. fit_and_plot_pytorch(0)

输出:

  1. L2 norm of w: 12.86785888671875

3.12_output3.png

  1. fit_and_plot_pytorch(3)

输出:

  1. L2 norm of w: 0.09631537646055222

3.12_output4.png

小结

  • 正则化通过为模型损失函数添加惩罚项使学出的模型参数值较小,是应对过拟合的常用手段。
  • 权重衰减等价于3.12 权重衰减 - 图45范数正则化,通常会使学到的权重参数的元素较接近0。
  • 权重衰减可以通过优化器中的weight_decay超参数来指定。
  • 可以定义多个优化器实例对不同的模型参数使用不同的迭代方法。

注:本节除了代码之外与原书基本相同,原书传送门