当前位置:   article > 正文

pytorch 实现线性回归 softmax(Pytorch 04)

pytorch 实现线性回归 softmax(Pytorch 04)

一 softmax 定义

softmax 是多分类问题,对决策结果不是多少,而是分类,哪一个。

为了估计所有可能类别的条件概率,我们需要一个有 多个输出的模型,每个类别对应一个输出。为了解决线 性模型的分类问题,我们需要和输出一样多的仿射函数(affine function)。每个输出对应于它自己的仿射函数。在我们的例子中,由于我们有 4个特征和3个可能的输出类别,我们将需要12个标量来表示权重(带下标 的w),3个标量来表示偏置(带下标的b)。

与线性回归一样,softmax回归也是一个 单层神经网络。 由于计算每个输出o1、o2和o3取决于所有输入x1、x2、x3和x4,所以softmax回归的 输出层也是全连接层

现在我们将优化参数以最大化观测数据的概率。为了得到预测结果,我们将设置一个阈值,如 选择具有最大概率的标签

要将输出视为概率,我们必须 保证在任何数据上的输出都是非负的且总和为1。此外,我们需要一个训练的目标函数,来激励模型精准地估计概率。

尽管softmax是一个非线性函数,但softmax回归的输出仍然由输入特征的仿射变换决定。因此,softmax回归是一个线性模型(linear model)。

1.1 交叉熵损失:

导数是我们softmax模型分配的概率与实际发生的情况(由独热标签向量表示)之间的差异。从 这个意义上讲,这与我们在回归中看到的非常相似,其中 梯度是观测值y和估计值yˆ之间的差异。这不是巧合, 在任何指数族分布模型中,对数似然的梯度正是由此得出的。这使 梯度计算在实践中变得容易很多。对于标签y,我们可以使用与以前相 同的表示形式。唯一的区别是,我们现在用一个 概率向量 表示,如(0.1, 0.2, 0.7),而不是仅包含二元项的向 量(0, 0, 1)。我们使用下公式来定义损失l,它是所有标签分布的预期损失值。此损失称为交叉熵损失(cross‐ entropy loss),它是分类问题最常用的损失之一。

1.2 模型预测和评估

在训练softmax回归模型后,给出任何样本特征,我们可以 预测每个输出类别的概率。通常我们 使用预测概率最高的类别作为输出类别。如果预测与实际类别(标签)一致,则预测是正确的。在接下来的实验中,我 们将使用精度(accuracy)来评估模型的性能。精度等于正确预测数与预测总数之间的比率。

  • softmax运算获取一个向量并将其映射为概率。
  • softmax回归适用于分类问题,它使用了softmax运算中输出类别的概率分布。

二 MNIST数据集 导入

Fashion‐MNIST由 10个类别的图像 组成,每个类别由训练数据集(train dataset)中的6000张图像和测试数据 集(test dataset)中的1000张图像组成。

  1. %matplotlib inline
  2. import torch
  3. import torchvision
  4. from torch.utils import data
  5. from torchvision import transforms
  6. from d2l import torch as d2l
  7. d2l.use_svg_display()
  1. # 通过ToTensor实例将图像数据从PIL类型变换成32位浮点数格式,
  2. # 并除以255使得所有像素的数值均在0~1之间
  3. trans = transforms.ToTensor() # download=True
  4. mnist_train = torchvision.datasets.FashionMNIST(root="../data", train=True,
  5. transform=trans, download=False)
  6. mnist_test = torchvision.datasets.FashionMNIST(root="../data", train=False,
  7. transform=trans, download=False)
  8. len(mnist_train), len(mnist_test)
  9. # (60000, 10000)

每个输入图像的高度和宽度均为28像素。数据集由灰度图像组成,其通道数为1。

  1. mnist_train[0][0].shape
  2. # torch.Size([1, 28, 28])

Fashion‐MNIST中包含的10个类别,分别为 t‐shirt(T恤)、trouser(裤子)、pullover(套衫)、dress(连衣 裙)、coat(外套)、sandal(凉鞋)、shirt(衬衫)、sneaker(运动鞋)、bag(包)和ankle boot(短靴)

  1. def get_fashion_mnist_labels(labels): #@save
  2. """返回Fashion-MNIST数据集的文本标签"""
  3. text_labels = ['t-shirt', 'trouser', 'pullover', 'dress', 'coat',
  4. 'sandal', 'shirt', 'sneaker', 'bag', 'ankle boot']
  5. return [text_labels[int(i)] for i in labels]
  6. def show_images(imgs, num_rows, num_cols, titles=None, scale=1.5): #@save
  7. """绘制图像列表"""
  8. figsize = (num_cols * scale, num_rows * scale)
  9. _, axes = d2l.plt.subplots(num_rows, num_cols, figsize=figsize)
  10. axes = axes.flatten()
  11. for i, (ax, img) in enumerate(zip(axes, imgs)):
  12. if torch.is_tensor(img):
  13. # 图片张量
  14. ax.imshow(img.numpy())
  15. else:
  16. # PIL图片
  17. ax.imshow(img)
  18. ax.axes.get_xaxis().set_visible(False)
  19. ax.axes.get_yaxis().set_visible(False)
  20. if titles:
  21. ax.set_title(titles[i])
  22. return axes
  23. X, y = next(iter(data.DataLoader(mnist_train, batch_size=18)))
  24. show_images(X.reshape(18, 28, 28), 2, 9, titles=get_fashion_mnist_labels(y));

 

 三 从零开始实现 softmax

3.1 读取 MNIST数据集

  1. def get_dataloader_workers(): #@save
  2. """使用4个进程来读取数据"""
  3. return 4
  4. def load_data_fashion_mnist(batch_size, resize=None): #@save
  5. """下载Fashion-MNIST数据集,然后将其加载到内存中"""
  6. trans = [transforms.ToTensor()]
  7. if resize:
  8. trans.insert(0, transforms.Resize(resize))
  9. trans = transforms.Compose(trans)
  10. mnist_train = torchvision.datasets.FashionMNIST(
  11. root="../data", train=True, transform=trans, download=True)
  12. mnist_test = torchvision.datasets.FashionMNIST(
  13. root="../data", train=False, transform=trans, download=True)
  14. return (data.DataLoader(mnist_train, batch_size, shuffle=True,
  15. num_workers=get_dataloader_workers()),
  16. data.DataLoader(mnist_test, batch_size, shuffle=False,
  17. num_workers=get_dataloader_workers()))
  1. import torch
  2. from IPython import display
  3. from d2l import torch as d2l
  4. batch_size = 256
  5. train_iter, test_iter = d2l.load_data_fashion_mnist(batch_size)
  6. train_iter, test_iter
  7. # (<torch.utils.data.dataloader.DataLoader at 0x27e6143bd30>,
  8. # <torch.utils.data.dataloader.DataLoader at 0x27e61e1a970>)

3.2 初始化模型参数

  1. num_inputs = 784
  2. num_outputs = 10
  3. W = torch.normal(0, 0.01, size=(num_inputs, num_outputs), requires_grad=True)
  4. b = torch.zeros(num_outputs, requires_grad=True)
  5. W, b
  6. # (tensor([[ 7.8446e-03, 3.8895e-04, 7.4652e-03, ..., 7.9335e-03,
  7. # -2.6370e-02, -3.4869e-03],
  8. # [ 1.1907e-02, -1.7130e-03, -5.7840e-04, ..., 1.7916e-04,
  9. # -5.7439e-03, 9.6542e-03],
  10. # [ 3.0170e-02, -1.4055e-02, 1.8777e-02, ..., 8.5911e-03,
  11. # 4.6043e-03, 3.0010e-03],
  12. # ...,
  13. # [-1.2875e-03, -8.0845e-03, -3.4810e-02, ..., 1.0136e-02,
  14. # -1.7731e-02, 3.4934e-03],
  15. # [-3.7752e-03, -6.9249e-03, 9.0967e-04, ..., 1.6938e-02,
  16. # 1.4804e-02, 8.6243e-03],
  17. # [ 1.7685e-02, -6.8463e-03, -4.2527e-05, ..., 5.0289e-03,
  18. # -9.5934e-03, -6.3647e-03]], requires_grad=True),
  19. # tensor([0., 0., 0., 0., 0., 0., 0., 0., 0., 0.], requires_grad=True))

3.3 定义softmax操作

给定一个矩阵X,我们可以对所有元素求和(默认情况下)。也可以 只求同一个轴上的元素,即 同一列(轴0)或同一行(轴1)。如果X是一个形状为(2, 3)的张量,我们 对列进行求和,则结果将是一个具 有形状(3,)的向量。当调用sum运算符时,我们可以指定保持在原始张量的轴数,而不折叠求和的维度。这将 产生一个具有形状(1, 3)的二维张量。

  1. X = torch.tensor([[1.0, 2.0, 3.0], [4.0, 5.0, 6.0]])
  2. X.sum(0, keepdim=True), X.sum(1, keepdim=True)
  3. # (tensor([[5., 7., 9.]]),
  4. # tensor([[ 6.],
  5. # [15.]]))

3.4 定义模型

定义了输入如何通过网络映射到输出。注 意,将数据传递到模型之前,我们使用 reshape 函数将每张原始图像 展平为向量

  1. def softmax(X):
  2. X_exp = torch.exp(X)
  3. partition = X_exp.sum(1, keepdim=True)
  4. return X_exp / partition # 这里应用了广播机制
  1. X = torch.normal(0, 1, (2, 5))
  2. X_prob = softmax(X)
  3. X_prob, X_prob.sum(1)
  4. # (tensor([[0.0589, 0.1685, 0.4852, 0.0695, 0.2180],
  5. # [0.7117, 0.0458, 0.0469, 0.1268, 0.0688]]),
  6. # tensor([1., 1.]))
  1. def net(X):
  2. return softmax(torch.matmul(X.reshape((-1, W.shape[0])), W) + b)

 

3.5 定义损失

实现引入的 交叉熵损失函数。这可能是深度学习中 最常见的损失函数,因为目前 分类问题的数量远远超过回归问题的数量。 回顾一下,交叉熵采用真实标签的预测概率的负对数似然。

  1. y = torch.tensor([0, 2])
  2. y_hat = torch.tensor([[0.1, 0.3, 0.6], [0.3, 0.2, 0.5]])
  3. y_hat[[0, 1], y]
  4. # tensor([0.1000, 0.5000])
  1. def cross_entropy(y_hat, y):
  2. return - torch.log(y_hat[range(len(y_hat)), y])
  3. cross_entropy(y_hat, y)
  4. # tensor([2.3026, 0.6931])

给定预测概率分布y_hat,当我们必须输出硬预测(hard prediction)时,我们 通常选择预测概率最高的类。 许多应用都要求我们做出选择。如Gmail必须将电子邮件分类为“Primary(主要邮件)”、“Social(社交邮 件)”“Updates(更新邮件)”或“Forums(论坛邮件)”。Gmail做分类时可能在内部估计概率,但最终它必 须在类中选择一个。

当 预测与标签分类y一致时,即是正确的。分类精度即正确预测数量与总预测数量之比。

为了计算精度,我们执行以下操作。首先,如果y_hat是矩阵,那么假定第二个维度存储每个类的预测分数。 我们使用argmax获得每行中最大元素的索引来获得预测类别。然后我们将预测类别与真实y元素进行比较。由 于等式运算符“==”对数据类型很敏感,因此我们 将 y_hat 的数据类型转换为与y的数据类型一致。结果是一 个包含0(错)和1(对)的张量

  1. def accuracy(y_hat, y): #@save
  2. """计算预测正确的数量"""
  3. if len(y_hat.shape) > 1 and y_hat.shape[1] > 1:
  4. y_hat = y_hat.argmax(axis=1)
  5. cmp = y_hat.type(y.dtype) == y
  6. return float(cmp.type(y.dtype).sum())
  1. accuracy(y_hat, y) / len(y)
  2. # 0.5

对于任意 数据迭代器 data_iter 可访问的数据集,我们可以评估在任意模型net的精度。

  1. def evaluate_accuracy(net, data_iter): #@save
  2. """计算在指定数据集上模型的精度"""
  3. if isinstance(net, torch.nn.Module):
  4. net.eval() # 将模型设置为评估模式
  5. metric = Accumulator(2) # 正确预测数、预测总数
  6. with torch.no_grad():
  7. for X, y in data_iter:
  8. metric.add(accuracy(net(X), y), y.numel())
  9. return metric[0] / metric[1]

这里定义一个实用程序类Accumulator,用于 对多个变量进行累加。在上面的evaluate_accuracy函数中,我 们在Accumulator实例中创建了2个变量,分别用于 存储正确预测的数量和预测的总数量。当我们遍历数据集 时,两者都将随着时间的推移而累加。

  1. class Accumulator: #@save
  2. """在n个变量上累加"""
  3. def __init__(self, n):
  4. self.data = [0.0] * n
  5. def add(self, *args):
  6. self.data = [a + float(b) for a, b in zip(self.data, args)]
  7. def reset(self):
  8. self.data = [0.0] * len(self.data)
  9. def __getitem__(self, idx):
  10. return self.data[idx]

由于我们使用随机权重初始化net模型,因此该模型的精度应接近于随机猜测。例如在 有10个类别情况下的 精度为0.1

  1. evaluate_accuracy(net, test_iter)
  2. # 0.1045

3.6 执行训练

  1. def train_epoch_ch3(net, train_iter, loss, updater): #@save
  2. """训练模型一个迭代周期(定义见第3章)"""
  3. # 将模型设置为训练模式
  4. if isinstance(net, torch.nn.Module):
  5. net.train()
  6. # 训练损失总和、训练准确度总和、样本数
  7. metric = Accumulator(3)
  8. for X, y in train_iter:
  9. # 计算梯度并更新参数
  10. y_hat = net(X)
  11. l = loss(y_hat, y)
  12. if isinstance(updater, torch.optim.Optimizer):
  13. # 使用PyTorch内置的优化器和损失函数
  14. updater.zero_grad()
  15. l.mean().backward()
  16. updater.step()
  17. else:
  18. # 使用定制的优化器和损失函数
  19. l.sum().backward()
  20. updater(X.shape[0])
  21. metric.add(float(l.sum()), accuracy(y_hat, y), y.numel())
  22. # 返回训练损失和训练精度
  23. return metric[0] / metric[2], metric[1] / metric[2]

在展示训练函数的实现之前,我们定义一个在 动画中绘制 数据的实用程序类Animator:

  1. class Animator: #@save
  2. """在动画中绘制数据"""
  3. def __init__(self, xlabel=None, ylabel=None, legend=None, xlim=None,
  4. ylim=None, xscale='linear', yscale='linear',
  5. fmts=('-', 'm--', 'g-.', 'r:'), nrows=1, ncols=1,
  6. figsize=(3.5, 2.5)):
  7. # 增量地绘制多条线
  8. if legend is None:
  9. legend = []
  10. d2l.use_svg_display()
  11. self.fig, self.axes = d2l.plt.subplots(nrows, ncols, figsize=figsize)
  12. if nrows * ncols == 1:
  13. self.axes = [self.axes, ]
  14. # 使用lambda函数捕获参数
  15. self.config_axes = lambda: d2l.set_axes(
  16. self.axes[0], xlabel, ylabel, xlim, ylim, xscale, yscale, legend)
  17. self.X, self.Y, self.fmts = None, None, fmts
  18. def add(self, x, y):
  19. # 向图表中添加多个数据点
  20. if not hasattr(y, "__len__"):
  21. y = [y]
  22. n = len(y)
  23. if not hasattr(x, "__len__"):
  24. x = [x] * n
  25. if not self.X:
  26. self.X = [[] for _ in range(n)]
  27. if not self.Y:
  28. self.Y = [[] for _ in range(n)]
  29. for i, (a, b) in enumerate(zip(x, y)):
  30. if a is not None and b is not None:
  31. self.X[i].append(a)
  32. self.Y[i].append(b)
  33. self.axes[0].cla()
  34. for x, y, fmt in zip(self.X, self.Y, self.fmts):
  35. self.axes[0].plot(x, y, fmt)
  36. self.config_axes()
  37. display.display(self.fig)
  38. display.clear_output(wait=True)

接下来我们实现一个训练函数,它会在 train_iter访问到的训练数据集上训练一个模型net。该训练函数将 会运行多个迭代周期(由num_epochs指定)。在每个迭代周期结束时,利用test_iter访问到的测试数据集对 模型进行评估。我们将利用Animator类来可视化训练进度。

  1. def train_ch3(net, train_iter, test_iter, loss, num_epochs, updater): #@save
  2. """训练模型(定义见第3章)"""
  3. animator = Animator(xlabel='epoch', xlim=[1, num_epochs], ylim=[0.3, 0.9],
  4. legend=['train loss', 'train acc', 'test acc'])
  5. for epoch in range(num_epochs):
  6. train_metrics = train_epoch_ch3(net, train_iter, loss, updater)
  7. test_acc = evaluate_accuracy(net, test_iter)
  8. animator.add(epoch + 1, train_metrics + (test_acc,))
  9. train_loss, train_acc = train_metrics
  10. assert train_loss < 0.5, train_loss
  11. assert train_acc <= 1 and train_acc > 0.7, train_acc
  12. assert test_acc <= 1 and test_acc > 0.7, test_acc

作为一个从零开始的实现,我们使用定义的 小批量随机梯度下降 来优化模型的损失函数,设置学习 率为0.1。

  1. lr = 0.1
  2. def updater(batch_size):
  3. return d2l.sgd([W, b], lr, batch_size)

我们训练模型 10个迭代周期迭代周期(num_epochs)和学习率(lr)都是可调节的超参数。 通过更改它们的值,我们可以提高模型的分类精度。

  1. num_epochs = 10
  2. train_ch3(net, train_iter, test_iter, cross_entropy, num_epochs, updater)

3.7 执行训练

  1. def predict_ch3(net, test_iter, n=6): #@save
  2. """预测标签(定义见第3章)"""
  3. for X, y in test_iter:
  4. break
  5. trues = d2l.get_fashion_mnist_labels(y)
  6. preds = d2l.get_fashion_mnist_labels(net(X).argmax(axis=1))
  7. titles = [true +'\n' + pred for true, pred in zip(trues, preds)]
  8. d2l.show_images( X[0:n].reshape((n, 28, 28)), 1, n, titles=titles[0:n])
  9. predict_ch3(net, test_iter)

训练过程:先读取数据,再定义模型和损失函数,然后 使用优化算法训练模型。大多数常见的深度学习模型都有类似的训练过程。

四 导包实现

  1. import torch
  2. from torch import nn
  3. from d2l import torch as d2l

 读取数据集:

  1. batch_size = 256
  2. train_iter, test_iter = d2l.load_data_fashion_mnist(batch_size)
  3. len(train_iter), len(test_iter)
  4. # (235, 40)

初始化权重:

  1. # PyTorch不会隐式地调整输入的形状。因此,
  2. # 我们在线性层前定义了展平层(flatten),来调整网络输入的形状
  3. net = nn.Sequential(nn.Flatten(), nn.Linear(784, 10))
  4. def init_weights(m):
  5. if type(m) == nn.Linear:
  6. nn.init.normal_(m.weight, std=0.01)
  7. net.apply(init_weights)
  8. # Sequential(
  9. # (0): Flatten(start_dim=1, end_dim=-1)
  10. # (1): Linear(in_features=784, out_features=10, bias=True)
  11. # )

定义 交叉熵损失损失:

  1. loss = nn.CrossEntropyLoss(reduction='none')
  2. loss
  3. # CrossEntropyLoss()

梯度转换:

  1. trainer = torch.optim.SGD(net.parameters(), lr = 0.1)
  2. trainer
  3. # SGD (
  4. # Parameter Group 0
  5. # dampening: 0
  6. # differentiable: False
  7. # foreach: None
  8. # lr: 0.1
  9. # maximize: False
  10. # momentum: 0
  11. # nesterov: False
  12. # weight_decay: 0
  13. # )

执行训练:

  1. num_epochs = 10
  2. d2l.train_ch3(net, train_iter, test_iter, loss, num_epochs, trainer)

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Cpp五条/article/detail/334907
推荐阅读
相关标签
  

闽ICP备14008679号