【深度学习】softmax回归的简洁实现

embedded/2025/2/8 2:34:17/

softmax回归的简洁实现

我们发现(通过深度学习框架的高级API能够使实现)(softmax)线性(回归变得更加容易)。
同样,通过深度学习框架的高级API也能更方便地实现softmax回归模型。
本节继续使用Fashion-MNIST数据集,并保持批量大小为256。

import torch
from torch import nn
from d2l import torch as d2l

初始化模型参数

[softmax回归的输出层是一个全连接层]。
因此,为了实现我们的模型,我们只需在Sequential中添加一个带有10个输出的全连接层。
同样,在这里Sequential并不是必要的,但它是实现深度模型的基础。
我们仍然以均值0和标准差0.01随机初始化权重。

'''
PyTorch不会隐式地调整输入的形状。
因此,我们在线性层前定义了展平层(flatten),来调整网络输入的形状
'''
net = nn.Sequential(nn.Flatten(), nn.Linear(784, 10))def init_weights(m):if type(m) == nn.Linear:nn.init.normal_(m.weight, std=0.01)net.apply(init_weights);
  • nn.Sequential
    这是 PyTorch 里的一个容器模块,
    其功能是按顺序依次排列多个神经网络层。
    在执行前向传播时,输入数据会依照层的先后顺序依次通过各个层。
  • nn.Flatten()
    该层的主要作用是把输入的多维张量展平为一维张量。
    方便后续输入到全连接层。
  • nn.Linear(784, 10)
    这是一个全连接层(线性层)。
    全连接层会对输入的 784 维向量进行线性变换,输出一个 10 维的向量。
  • net.apply(init_weights)
    applynn.Module 类的一个方法,它会递归地把指定的函数(这里是 init_weights)应用到 net 网络的每一个子模块上。也就是说,对于 net 中的每个子层,都会调用 init_weights 函数进行权重初始化。

重新审视Softmax的实现

在前面例子中,我们计算了模型的输出,然后将此输出送入交叉熵损失。

从数学上讲,这是一件完全合理的事情。

然而,从计算角度来看,指数可能会造成数值稳定性问题。

回想一下,
softmax函数 y ^ j = exp ⁡ ( o j ) ∑ k exp ⁡ ( o k ) \hat y_j = \frac{\exp(o_j)}{\sum_k \exp(o_k)} y^j=kexp(ok)exp(oj)
其中 y ^ j \hat y_j y^j是预测的概率分布。 o j o_j oj是未规范化的预测 o \mathbf{o} o的第 j j j个元素。如果 o k o_k ok中的一些数值非常大,那么 exp ⁡ ( o k ) \exp(o_k) exp(ok)可能大于数据类型容许的最大数字,即上溢(overflow)。
这将使分母或分子变为inf(无穷大),
最后得到的是0、infnan(不是数字)的 y ^ j \hat y_j y^j
在这些情况下,我们无法得到一个明确定义的交叉熵值。

解决这个问题的一个技巧是:
在继续softmax计算之前,先从所有 o k o_k ok中减去 max ⁡ ( o k ) \max(o_k) max(ok)
这里可以看到每个 o k o_k ok按常数进行的移动不会改变softmax的返回值:

y ^ j = exp ⁡ ( o j − max ⁡ ( o k ) ) exp ⁡ ( max ⁡ ( o k ) ) ∑ k exp ⁡ ( o k − max ⁡ ( o k ) ) exp ⁡ ( max ⁡ ( o k ) ) = exp ⁡ ( o j − max ⁡ ( o k ) ) ∑ k exp ⁡ ( o k − max ⁡ ( o k ) ) . \begin{aligned} \hat y_j & = \frac{\exp(o_j - \max(o_k))\exp(\max(o_k))}{\sum_k \exp(o_k - \max(o_k))\exp(\max(o_k))} \\ & = \frac{\exp(o_j - \max(o_k))}{\sum_k \exp(o_k - \max(o_k))}. \end{aligned} y^j=kexp(okmax(ok))exp(max(ok))exp(ojmax(ok))exp(max(ok))=kexp(okmax(ok))exp(ojmax(ok)).

在减法和规范化步骤之后,可能有些 o j − max ⁡ ( o k ) o_j - \max(o_k) ojmax(ok)具有较大的负值。
由于精度受限, exp ⁡ ( o j − max ⁡ ( o k ) ) \exp(o_j - \max(o_k)) exp(ojmax(ok))将有接近零的值,即下溢(underflow)。
这些值可能会四舍五入为零,使 y ^ j \hat y_j y^j为零,
并且使得 log ⁡ ( y ^ j ) \log(\hat y_j) log(y^j)的值为-inf
反向传播几步后,我们可能会发现自己面对一屏幕可怕的nan结果。

尽管我们要计算指数函数,但我们最终在计算交叉熵损失时会取它们的对数。
通过将softmax和交叉熵结合在一起,可以避免反向传播过程中可能会困扰我们的数值稳定性问题。
如下面的等式所示,我们避免计算 exp ⁡ ( o j − max ⁡ ( o k ) ) \exp(o_j - \max(o_k)) exp(ojmax(ok))
而可以直接使用 o j − max ⁡ ( o k ) o_j - \max(o_k) ojmax(ok),因为 log ⁡ ( exp ⁡ ( ⋅ ) ) \log(\exp(\cdot)) log(exp())被抵消了。

log ⁡ ( y ^ j ) = log ⁡ ( exp ⁡ ( o j − max ⁡ ( o k ) ) ∑ k exp ⁡ ( o k − max ⁡ ( o k ) ) ) = log ⁡ ( exp ⁡ ( o j − max ⁡ ( o k ) ) ) − log ⁡ ( ∑ k exp ⁡ ( o k − max ⁡ ( o k ) ) ) = o j − max ⁡ ( o k ) − log ⁡ ( ∑ k exp ⁡ ( o k − max ⁡ ( o k ) ) ) . \begin{aligned} \log{(\hat y_j)} & = \log\left( \frac{\exp(o_j - \max(o_k))}{\sum_k \exp(o_k - \max(o_k))}\right) \\ & = \log{(\exp(o_j - \max(o_k)))}-\log{\left( \sum_k \exp(o_k - \max(o_k)) \right)} \\ & = o_j - \max(o_k) -\log{\left( \sum_k \exp(o_k - \max(o_k)) \right)}. \end{aligned} log(y^j)=log(kexp(okmax(ok))exp(ojmax(ok)))=log(exp(ojmax(ok)))log(kexp(okmax(ok)))=ojmax(ok)log(kexp(okmax(ok))).

我们也希望保留传统的softmax函数,以备我们需要评估通过模型输出的概率。
但是,我们没有将softmax概率传递到损失函数中,
而是[在交叉熵损失函数中传递未规范化的预测,并同时计算softmax及其对数],
这是一种类似"LogSumExp技巧"的聪明方式。

loss = nn.CrossEntropyLoss(reduction='none')
  • nn.CrossEntropyLoss
    nn.CrossEntropyLoss 是 PyTorch 中用于计算交叉熵损失的类。
    在分类问题中,它结合了 nn.LogSoftmax()nn.NLLLoss() 两个操作,适用于多分类任务。其输入通常是模型的原始输出(未经过 Softmax 激活函数处理)和真实标签。
  • reduction=‘none’
    reductionnn.CrossEntropyLoss 类的一个重要参数,它控制着如何对每个样本的损失进行汇总,具体有以下几种取值:
    • 'none':不进行任何汇总操作,直接返回每个样本的损失值,返回的结果是一个与输入样本数量相同的张量。
    • 'mean':(默认值)对每个样本的损失求平均值,返回一个标量值。
    • 'sum':对每个样本的损失求和,返回一个标量值。

优化算法

在这里,我们(使用学习率为0.1的小批量随机梯度下降作为优化算法)。
这与我们在线性回归例子中的相同,这说明了优化器的普适性。

trainer = torch.optim.SGD(net.parameters(), lr=0.1)

net.parameters() 是一个生成器,它会返回模型中所有需要学习的参数(如权重和偏置)。这些参数会被传递给优化器,以便优化器在训练过程中对它们进行更新。
torch.optim.SGD 是 PyTorch 中实现随机梯度下降优化算法的类。它接受模型的参数和一些超参数作为输入,用于更新模型的参数。

训练

接下来我们[调用] 之前(定义的训练函数来训练模型)。

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

在这里插入图片描述


http://www.ppmy.cn/embedded/160435.html

相关文章

C#常用744单词

1.visual 可见的 2.studio 工作室 3.dot 点 4.net 网 5.harp 尖端的,锋利的。 6.amework 骨架,构架,框架 7.beta 测试版,试用版 8.XML(全称:eXtensible Markup Language&#xff09…

RabbitMQ 从入门到精通:从工作模式到集群部署实战(三)

文章目录 使用CLI管理RabbitMQrabbitmqctlrabbitmq-queuesrabbitmq-diagnosticsrabbitmq-pluginsrabbitmq-streamsrabbitmq-upgraderabbitmqadmin 使用CLI管理RabbitMQ RabbitMQ CLI 工具需要安装兼容的 Erlang/OTP版本。 这些工具假定系统区域设置为 UTF-8(例如en…

ASP.NET Core中间件的概念及基本使用

什么是中间件 中间件是ASP.NET Core的核心组件,MVC框架、响应缓存、身份验证、CORS、Swagger等都是内置中间件。 广义上来讲:Tomcat、WebLogic、Redis、IIS;狭义上来讲,ASP.NET Core中的中间件指ASP.NET Core中的一个组件。中间件…

深度学习 - 神经网络的原理

## 深度学习 - 神经网络的原理 深度学习是机器学习的一个分支,其核心是模拟人脑神经网络的结构和功能,构建多层的神经网络模型,从数据中学习特征并进行预测或分类。 **神经网络的基本原理:** 1. **神经元模型:** * 神经网…

扬帆数据结构算法之舟,启航C++探索征途——LeetCode深度磨砺:顺序表技术精进实践

人无完人,持之以恒,方能见真我!!! 共同进步!! 文章目录 顺序表练习1.移除数组中指定的元素方法1(顺序表)方法2(双指针) 2.删除有序数组中的重复项…

数据结构-基础

1、概念: 程序 数据结构 算法 2、程序的好坏 可读性,稳定性,扩展性,时间复杂度,空间复杂度。 3、数据结构 是指存储、组织数据的方式,以便高效地进行访问和修改。通过选择适当的数据结构, 能…

「全网最细 + 实战源码案例」设计模式——模板方法模式

核心思想 模板方法模式(Template Method Pattern)是一种行为型设计模式,定义了一个算法的骨架(模板),将某些步骤延迟到子类中实现(在不修改结构的情况下),以避免代码重复…

音频录制一般在什么情况下会选择保存为PCM?什么情况会选择保存为WAV?

在音频开发中,选择保存为 PCM 或 WAV 格式取决于具体的应用场景和需求。以下是两种格式的特点以及适用场景的分析: PCM 格式 特点: 原始音频数据: PCM 是未压缩的原始音频数据,没有任何文件头或元数据。数据直接以二进制形式存储,通常是采样点的值。文件体积较大: 由于…