- Java锁的逻辑(结合对象头和ObjectMonitor)
- 还在用饼状图?来瞧瞧这些炫酷的百分比可视化新图形(附代码实现)⛵
- 自动注册实体类到EntityFrameworkCore上下文,并适配ABP及ABPVNext
- 基于Sklearn机器学习代码实战
。
在各种深度学习框架中,我们最常用的损失函数就是交叉熵,熵是用来描述一个系统的混乱程度,通过交叉熵我们就能够确定预测数据与真实数据的相近程度。交叉熵越小,表示数据越接近真实样本.
二分类 。
单标签任务,顾名思义,每个样本只能有一个标签,比如ImageNet图像分类任务,或者MNIST手写数字识别数据集,每张图片只能有一个固定的标签。二分类是多分类任务中的一个特例,因为二分类只有正样本和负样本,并且两者的概率之和为1,所以不需要预测一个向量,只需要输出一个概率值就好了。损失函数一般是输出经过sigmoid激活函数之后,采用交叉熵损失函数计算loss.
以上面猫狗二分类任务为例,网络最后一层的输出应该理解为:网络认为图片中含有这一类别物体的概率。而每一类的真实标签都只有两种可能值,即“不是这一类物体”和“是这一类物体”,这是一个二项分布,可能的取值为0或者1,而网络预测的分布可以理解为标签是1的概率。当网络的输出logits=2时,经过sigmoid得到为狗的概率是0.9,交叉熵损失loss=-1×log(0.9) - 0×log(0.1) ≈ 0.1.
多分类 。
在多分类任务中,利用softmax函数将多个神经元(神经元数目为类别数)输出的结果映射到对于总输出的占比(范围0~1,占比可以理解成概率值),我们通过选择概率最大输出类别作为预测类别.
上面为三分类任务,输出的logits向量对应三个类别,经过softmax后得到三个和为1的概率[0.9,0.1,0]。样本“剪刀”对应的真实分布为[1,0,0],此时计算损失函数得loss = -1*log(0.9) - 0×log(0.1) - 0×log(0) ≈ 0.1。如果网络输出的概率为[0.1,0.9,0],此时的交叉熵损失为loss= -1*log(0.1) - 0×log(0.9) - 0×log(0)= 1。上述两种情况对比,第一个分布的损失明显低于第二个分布的损失,说明第一个分布更接近于真实分布,事实也确实是这样.
多标签分类任务,即 一个样本可以有多个标签 ,比如一张图片中同时含有“猫”和“狗”,这张图片就同时拥有属于“猫”和“狗”的两种标签。在这种情况下,我们将函数作为网络最后一层的输出,把网络最后一层的每个神经元都看做任务中的一个类别,以图像识别任务为例,网络最后一层的输出应该理解为:网络认为图片中含有这一类别物体的概率。而每一类的真实标签都只有两种可能值,即“图片中含有这一类物体”和“图片中不含有这一类物体”,这是一个二项分布。综上所述,对多分类任务中的每一类单独分析的话,真实分布是一个二项分布,可能的取值为0或者1,而网络预测的分布可以理解为标签是1的概率。此外,由于多标签分类任务中,每一类是相互独立的,所以网络最后一层神经元输出的概率值之和并不等于1.
上面的多标签分类任务有三个标签:狗,猫,猪。输入图片中没有猪,所以真实分布应该为:[ 1, 1, 0 ] .
假设经过右图的网络输出的概率分布为:[ 0.95, 0.73, 0.05],则我们可以对狗,猫,猪这三类都计算交叉熵损失函数,然后将它们相加就得到这一张图片样本的交叉熵损失函数值.
loss狗=-1×log(0.95)-(1-1)×log(1-0.95)≈0.05 。
loss猫=-1×log(0.73)-(1-1)×log(1-0.73)≈0.31 。
loss猪=-0×log(0.05)-(1-0)×log(1-0.05)≈0.05 。
loss总=loss狗+loss猫+loss猪=0.05+0.31+0.05=0.41 。
假设经过右图的网络输出的概率分布为:[ 0.3, 0.5, 0.7],交叉熵损失损失为 。
loss狗=-1×log(0.3)-(1-1)×log(1-0.3)≈1.2 。
loss猫=-1×log(0.5)-(1-1)×log(1-0.5)≈0.7 。
loss猪=-0×log(0.7)-(1-0)×log(1-0.7)≈1.2 。
loss总=loss狗+loss猫+loss猪=1.2+0.7+1.2=3.1 。
由上面两种情况也可以看出,预测分布越接近真实分布,交叉熵损失越小,预测分布越远离真实分布,交叉熵损失越大.
Pytorch关于损失函数的内容,可以在官方文档 torch.nn — PyTorch 1.10 documentation 里找到.
BCEloss主要用于计算 标签只有1或者0时的二分类损失 ,标签和预测值是一一对应的。需要注意的是,通过nn.BCEloss来计算损失前,需要对预测值进行一次sigmoid计算。sigmoid函数会将预测值映射到0-1之间。如果觉得手动加sigmoid函数麻烦,可以直接调用 nn.BCEwithlogitsloss .
# class torch.nn.BCELoss(weight=None, size_average=None, reduce=None, reduction='mean') # function torch.nn.functional.binary_cross_entropy(input, target, weight=None, size_average=None, reduce=None, reduction='mean')
input ( Tensor ) – 任意维度的张量 。
target ( Tensor ) – 和输入一样的shape,但值必须在0-1之间 。
weight ( Tensor , optional ) – 人为给定的权重 。
size_average ( bool , optional ) – 已弃用 。
reduce ( bool , optional ) – 已弃用 。
reduction ( str , optional ) – none :求 minibatch 中每个sample的loss值,不做归并; mean :对 minibatch 中所有sample 的loss值求平均; sum :对 minibatch 中所有sample的loss值求和.
当 reduction = none时, 。
其中N表示batch_size,若reduction不为none时, 。
示例 。
import numpy as np import torch import torch.nn.functional as F input = torch.Tensor([[0.6, 0.1], [0.3, 0.8]]) target = torch.Tensor([[0, 1], [1, 0]]) loss = F.binary_cross_entropy(input, target) # loss : tensor(1.5081) loss = torch.sum(-(target * torch.log(input) + (1 - target) * torch.log(1 - input))) / 4 # loss : tensor(1.5081) loss = -(np.log(0.4) + np.log(0.1) + np.log(0.3) + np.log(0.2)) / 4 # 1.5080716354070594 loss = F.binary_cross_entropy(torch.sigmoid(input), target) # loss : tensor(0.8518) loss = F.binary_cross_entropy_with_logits(input, target) # loss : tensor(0.8518)
使用神经网络模型时,调整输出层的单元数,当进行 n分类(n>2) 时,设置输出层的单元数为n,采用softmax损失函数(把输出层整体转换为0-1之间的概率分布)+多分类交叉熵损失。把标签转换为one-hot向量,每个样本的标签是一个n维向量,其所属类别位置为1,其余位置为0.
#CLASS torch.nn.CrossEntropyLoss(weight=None, size_average=None, ignore_index=- 100, reduce=None, reduction='mean', label_smoothing=0.0) #FUNCTION torch.nn.functional.cross_entropy(input, target, weight=None, size_average=None, ignore_index=- 100, reduce=None, reduction='mean', label_smoothing=0.0)
cross_entropy的pytorch实现
def cross_entropy(input, target, weight=None, size_average=None, ignore_index=-100, reduce=None, reduction='mean'): if size_average is not None or reduce is not None: reduction = _Reduction.legacy_get_string(size_average, reduce) return nll_loss(log_softmax(input, 1), target, weight, None, ignore_index, None, reduction)
可以看出 softmax + log + NLLloss = crossEntropyLoss .
多分类问题(分类种类为c个)在经过输出层的计算后,会产生c个输出x,softmax的作用就是将输出x转化为和为1的概率问题。它是二分类函数sigmoid在多分类上的推广,目的是将多分类的结果以概率的形式展现出来。其定义如下:
概率是非负且和为1的,因此softmax首先将模型的预测结果转化到指数函数上,这样 保证了概率的 非负性 。再将转换后的结果归一化处理, 使得各预测结果的 概率之和等于1 。比如三分类预测结果为[3,1,-3],求指数得[20.09,2.72,0.05],归一化后得[0.88,0.12,0].
当输入x中存在特别大的xi 时,exp(xi) 会变得很大,导致出现 上溢 的情况。 当输入x中每个元素都为特别小的负数时,分母 会变得很小,超出精度范围时向下取0,导致 下溢 .
log_softmax是指在softmax函数的基础上再进行一次log运算,当x再0-1之间时,log(x)值在负无穷到0之间.
其中zmax是输入z中的最大值, 对于任何一个zi,减去zmax后 , exp(zi-zmax)的最大值为1,所以不会发生上溢。而∑exp(zi-zmax)中至少有一项值为1, 避免了计算log(0) ,也 解决了下溢的情况.
NLLloss输入是一个 对数概率向量 和一个 目标标签 ,也就是将上面的输出中与label对应的那个值拿出来,去掉负号再求均值。 不用对label进行one_hot编码 ,因为nll_loss函数已经实现了类似one-hot过程:直接在log(softmax(input))矩阵中,取出每个样本的target值对应的下标位置(该位置在onehot中为1,其余位置在onehot中为0).
示例 NLLloss 。
import torch import torch.nn.functional as F # 1D input = torch.Tensor([[2, 3, 1], [3, 7, 9]]) target = torch.tensor([1, 2]) loss = F.nll_loss(input, target) #loss: tensor(-6.) # 2D input = torch.Tensor([[[2, 3], [1, 5]], [[3, 7], [1, 9]]]) target = torch.tensor([[1, 1], [0, 0]]) loss = F.nll_loss(input, target) tensor(-4.)
在一维时,nllloss对两个向量的操作为,将input中的向量,在target中对应的index取出,并取负号输出。target中为1,则取2,3,1中的第1位3,target第二位为2,则取出3,7,9的第2位9,将两数取平均(当 reduction='mean'时) 后加负号后输出。在二维时(输入是图片),loss=[(-3)+(-7)+(-1)+(-5)]/4 = -4.
损失函数中的weight参数用于 调节不同类别样本占比差异很大的现象 ,比如语义分割中,背景的像素比缺陷的像素多很多,在计算loss的时候两类别loss直接相加会导致模型对背景的过拟合。在分类中,ok的样本过多而ng样本过少,当它们的比值大于10的时候要考虑样本不平衡问题。假设有两类,标签类别为0, 1,所对应的样本数量为1000,10。在网络学习的过程中,假设预测出来的标签都是0(100000个样本),它的准确率为1000/1010 ≈ 0.99,将近100%,所以模型就会朝着拟合标签0的方向更新,导致对标签0的样本过拟合,对1类别的样本欠拟合,泛化能力很差.
如何解决?
cross_entropy函数中的weight参数可以在分类问题中给不同的类别不同的权重.
示例 cross_entropy函数中的weight参数 。
import torch import torch.nn.functional as F input = torch.Tensor([[[1, 2], [3, 5]], [[4, 7], [4, 6]]]) # torch.Size([2, 2, 2]) target = torch.tensor([[0, 0], [1, 1]]) # torch.Size([2, 2]) weight = torch.tensor([1.0, 9.0]) loss = F.cross_entropy(input, target) # tensor(1.7955) loss = F.cross_entropy(input, target, weight) # tensor(1.1617)
pytorch官方对weight给出的解释是“如果提供,则重复该操作以匹配输入张量形状”,也就是说给出weight参数后,会将其shape和input的shape相匹配。默认情况,也就是weight=None时,上述公式中的Wn=1;当weight!=None时,也就意味着我们 需要为每一个样本赋予权重Wi .
示例 binary_cross_entropy函数中的weight参数 。
import torch import torch.nn.functional as F input = torch.rand(3, 3) target = torch.rand(3, 3).random_(2) w = [0.1, 0.9] # 标签0和标签1的权重 weight = torch.zeros(target.shape) # 权重矩阵 for i in range(target.shape[0]): for j in range(target.shape[1]): weight[i][j] = w[int(target[i][j])] loss = F.binary_cross_entropy(input, target, weight=weight) """ # input tensor([[0.1531, 0.3302, 0.7537], [0.2200, 0.6875, 0.2268], [0.5109, 0.5873, 0.9275]]) # target tensor([[1., 0., 0.], [0., 0., 1.], [0., 1., 0.]]) # weight tensor([[0.9000, 0.1000, 0.1000], [0.1000, 0.1000, 0.9000], [0.1000, 0.9000, 0.1000]]) # loss tensor(0.4621) """
当语义分割任务是 二分类 时,有两种情况 (1)最后一个卷积层直接输出1通道的feature map,做 sigmoid 后用binary_cross_entropy函数计算损失(2)最后一个卷积层输出2channel的feature map,在通道维度做 softmax ,然后利用cross_entropy计算损失。 这两种方法哪一个更好?
首先我们先理论上证明一下二者没有本质上的区别 ,对于二分类而言(以输入x1 为例):
Sigmoid函数:
Softmax函数:
令(x1-x2)=z,和公式(1)完全相同,所以理论上来说两者是没有任何区别的.
代码: WZMIAOMIAO / deep-learning-for-image-processing / pytorch_segmentation / unet / 。
DRIVE数据集: https://pan.baidu.com/s/1Tjkrx2B9FgoJk0KviA-rDw 密码: 8no8 。
视频讲解: https://www.bilibili.com/video/BV1Vq4y127fB 。
使用Up主 霹雳吧啦Wz 的UNet代码测试,源代码输出2通道后进行softmax。对网络进行以下改动,将其改为输出1通道,并使用相同的评价指标.
损失部分 。
# 二通道损失计算 def criterion(inputs, target, loss_weight=None, num_classes: int = 2, dice: bool = False, ignore_index: int = -100): losses = {} for name, x in inputs.items(): # 忽略target中值为255的像素,255的像素是目标边缘或者padding填充 loss = nn.functional.cross_entropy(x, target, ignore_index=ignore_index, weight=loss_weight) losses[name] = loss if len(losses) == 1: return losses['out'] return losses['out'] + 0.5 * losses['aux'] # 一通道损失计算 def criterion(inputs, target, loss_weight=None, num_classes: int = 2, dice: bool = False, ignore_index: int = -100): losses = {} for name, x in inputs.items(): # 将不关心区域(255)置为0 roi_mask = torch.eq(target, ignore_index) target[roi_mask] = 0
# reshape后target和x维度相同 target = target.reshape(-1).float() x = x.reshape(-1).float() loss = nn.functional.binary_cross_entropy_with_logits(x, target) losses[name] = loss if len(losses) == 1: return losses['out'] return losses['out'] + 0.5 * losses['aux']
评价指标 。
#二通道 def evaluate(model, data_loader, device, num_classes): model.eval() metric_logger = utils.MetricLogger(delimiter=" ") header = 'Test:' with torch.no_grad(): for image, target in metric_logger.log_every(data_loader, 100, header): image, target = image.to(device), target.to(device) output0 = model(image)['out'] output1 = torch.softmax(output0, dim=1) output2 = output1.argmax(dim=1).float() #只计算前景的dice_coeff target = target.float() dice = dice_coeff(output2, target, ignore_index=255) return dice.item() #一通道 def evaluate(model, data_loader, device, num_classes): model.eval() dice = utils.DiceCoefficient(num_classes=num_classes, ignore_index=255) metric_logger = utils.MetricLogger(delimiter=" ") header = 'Test:' with torch.no_grad(): for image, target in metric_logger.log_every(data_loader, 100, header): image, target = image.to(device), target.to(device) output = model(image)['out'] # 输出的是0-1之间的概率 output[output > 0.5] = 1 output[output < 1] = 0 target = target.float() dice = dice_coeff(output, target, ignore_index=255)
predict 。
# 二通道 ... output = model(img.to(device)) prediction = output['out'].argmax(1).squeeze(0) prediction = prediction.to("cpu").numpy().astype(np.uint8) # 将前景对应的像素值改成255(白色) prediction[prediction == 1] = 255 # 将不感兴趣的区域像素设置成0(黑色) prediction[roi_img == 0] = 0 mask = Image.fromarray(prediction) # 一通道 ... output = model(img.to(device)) prediction = torch.sigmoid(output['out']).squeeze(0).squeeze(0) prediction = prediction.to("cpu").numpy() # 将前景对应的像素值改成255(白色) prediction[prediction > 0.5] = 1 prediction[prediction < 1] = 0 # 将不感兴趣的区域像素设置成0(黑色) prediction[roi_img == 0] = 0 mask = Image.fromarray((prediction*255).astype(np.uint8))
batch_size取16,不使用dice_loss,训练150epoch后,效果差不多.
。
。
。
。
参考 。
1. PyTorch master documentation 。
2. t orch的交叉熵损失函数(cross_entropy)计算 。
3. 带权重的损失函数nn.crossEntropyLoss中的weight使用 。
4. 一文搞懂F.binary_cross_entropy以及weight参数 。
5. Cross-entropy 和 Binary cross-entropy 。
6. 交叉熵损失函数(Cross Entropy Loss) 。
7. 二分类问题,应该选择sigmoid还是softmax?
最后此篇关于交叉熵损失CrossEntropyLoss的文章就讲到这里了,如果你想了解更多关于交叉熵损失CrossEntropyLoss的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我想使用单个(交叉)编译器来编译不同 ARM 调用约定的代码:因为我总是想使用浮点和 NEON 指令,所以我只想选择硬浮点调用约定或软浮点(softfp)调用约定。 我的编译器默认为硬浮点,但它支持我
假设我正在构建一个依赖于两个库的 java 应用程序:A 和 B。A 和 B 都依赖于库 C。管理 A 和 B 使用相同版本的最佳方法是什么所以他们不冲突?我正在使用 Gradle。 最佳答案 从 G
我想在按钮的文本上添加图像。如果我将图像添加为按钮的背景,它就会添加到文本下方。预期结果作为图像添加。请帮忙 更新:我需要以编程方式执行此操作。 最佳答案 在 XML 中, * 在代码中
我已经开始使用 CSS3 制作动画了。 我尝试创建一个动画汉堡菜单,但结果有点难看。顶部和底部的条向右平移一点。所以旋转动画不是很流畅和正确。 这是结果 => 这是我的代码: /* HTML */
给定一个具有2条相交曲线的图像,如下图所示,我如何使用opencv或python检测和区分2条曲线? (所以我需要2条单独的曲线) 最佳答案 您可以扫描每一列,并从连接的零件中识别出簇。 伪算法: l
我正在尝试在 redhat 集群(x86_64 主机)上设置 cross-mingw。我没有 root 访问权限,并且可用的 mingw 二进制文件不起作用(坏 glibc 版本等)。我正在阅读本教程
我正在尝试在javaFX中开发一个游戏,当两个图像相交时,分数将被更新,并且障碍物将不可见。但不幸的是,在游戏中分数不断更新。 我想我无法在游戏中正确地使图像不可见。 以下是相关类的完整代码: pac
pikastar dot com 是网站,当向下滚动它然后在导航菜单展开固定位置时它 > 将穿过主 div。我该如何修复它。 #topNav.sticky { box-shadow: 0 10
我正在使用 Eclipse为 ARM 处理器交叉编译 g++ 项目。我在 Windows 环境中使用 yagarto 工具链。我对 C 项目没有问题,但是对于 C++,我一直收到错误: libc.a(
我想从两个哈希数组中获取并集/交集/差集,例如: array1 = [{:name =>'Guy1', :age => 45},{:name =>'Guy2', :age => 45}] array2
有没有办法在调用任何 Controller 操作之前执行一些代码? 我需要根据 get 参数的值设置 session 变量,而不考虑调用哪个 Controller 。 当然,一旦这个处理完成,请求需要
我刚开始使用 3D 网格,面向用于有限元分析。我想在立方体状矩阵中模拟 Material 的夹杂物(任何形状,但主要对球体和椭圆体感兴趣)。这些夹杂物不应彼此重合。 所以我想为python使用某种包,
我想知道以跨平台方式操作应用程序设置的最佳解决方案是什么。 在 iOS 中,我们可以在设置屏幕中更改应用程序外部的设置,但在 windows phone 和 android 中我们没有。 所以,我的想
var barcodeNum = ko.observable(""); VelocityMeetings.scan = function (params) { var errorMessage = k
这个问题在这里已经有了答案: Transforming data.frame in R (2 个答案) 关闭10 年前。 过去我问过一个关于如何create cross tables from a
我有两个共享同一个工厂的 Controller 。其中一个 Controller 正在更新工厂变量。其他人应该注意该变化并稍后显示。 我是这样尝试的: http://plnkr.co/edit/q1N
标题不好,但这是我发现的将我的问题与简单的表格交叉区分开来的方式,因为我之前的研究总是让我接触到这类主题。 我有几个表 - 为了简化起见,我们只用 3 个表来命名它们:A、B、C。我想将它们全部放在一
我需要做这样的事情(在 MySQL 中),我使用 UNION 的尝试直到现在才奏效。 理论上: SELECT * FROM tableA A JOIN tableB B ON A.tableAId =
注意:使用SDL 2.0,Cross header class问题 我在类之间进行交叉引用,主要是我的类初始化渲染器和我的纹理类引用渲染初始化。现在,我已经能够运行该程序,直到我开始放入纹理类,代码也
我有一个这样的字母数组 var letters = ["Y", "X", "A", "Y", "O", "H", "A", "O", "O"]; 我创建了一个循环来
我是一名优秀的程序员,十分优秀!