损失函数:DIOU loss手写实现

2023-10-28 21:59

本文主要是介绍损失函数:DIOU loss手写实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

下面是纯diou代码

            '''计算两个box的中心点距离d'''# d = math.sqrt((x2 - x1) ** 2 + (y2 - y1) ** 2)d = math.sqrt((pred[:, -1] - target[:, -1]) ** 2 + (pred[:, -2] - target[:, -2]) ** 2)# 左边xpred_l = pred[:, -1] - pred[:, -1] / 2target_l = target[:, -1] - target[:, -1] / 2# 上边ypred_t = pred[:, -2] - pred[:, -2] / 2target_t = target[:, -2] - target[:, -2] / 2# 右边xpred_r = pred[:, -1] + pred[:, -1] / 2target_r = target[:, -1] + target[:, -1] / 2# 下边ypred_b = pred[:, -2] + pred[:, -2] / 2target_b = target[:, -2] + target[:, -2] / 2'''计算两个box的bound的对角线距离'''bound_l = torch.min(pred_l, target_l)  # leftbound_r = torch.max(pred_r, target_r)  # rightbound_t = torch.min(pred_t, target_t)  # topbound_b = torch.max(pred_b, target_b)  # bottomc = math.sqrt((bound_r - bound_l) ** 2 + (bound_b - bound_t) ** 2)dloss = iou - (d ** 2) / (c ** 2)loss = 1 - dloss.clamp(min=-1.0, max=1.0)

第一步 计算两个box的中心点距离d

首先要知道pred和target的输出结果是什么
pred[:,:2]第一个:表示多个图片,第二个:2表示前两个数值,代表矩形框中心点(Y,X)
pred[:,2:]第一个:表示多个图片,第二个2:表示两个数值,代表矩形框长宽(H,W)
target[:,:2]同理,
d =
 

根据上面的分析来计算左右上下坐标lrtb

 然后计算内部2个矩形的最小外接矩形的对角线长度c

 d是两个预测矩形中心点的距离

 下面接受各种极端情况
A 两个框中心对齐时候,d/c=0,iou可能0-1

 A 两个框相距很远时,d/c=1,iou=0

 所以d/c属于0-1
dloss=iou-d/c属于-1到1
因此设置loss=1-dloss属于0-2

 

展示iou\giou\diou代码,这是YOLOX自带的损失函数,其中dloss是我自己写的
YOLOX是下载自
GitHub - Megvii-BaseDetection/YOLOX: YOLOX is a high-performance anchor-free YOLO, exceeding yolov3~v5 with MegEngine, ONNX, TensorRT, ncnn, and OpenVINO supported. Documentation: https://yolox.readthedocs.io/YOLOX is a high-performance anchor-free YOLO, exceeding yolov3~v5 with MegEngine, ONNX, TensorRT, ncnn, and OpenVINO supported. Documentation: https://yolox.readthedocs.io/ - GitHub - Megvii-BaseDetection/YOLOX: YOLOX is a high-performance anchor-free YOLO, exceeding yolov3~v5 with MegEngine, ONNX, TensorRT, ncnn, and OpenVINO supported. Documentation: https://yolox.readthedocs.io/https://github.com/Megvii-BaseDetection/YOLOX

class IOUloss(nn.Module):def __init__(self, reduction="none", loss_type="iou"):super(IOUloss, self).__init__()self.reduction = reductionself.loss_type = loss_typedef forward(self, pred, target):assert pred.shape[0] == target.shape[0]pred = pred.view(-1, 4)target = target.view(-1, 4)tl = torch.max((pred[:, :2] - pred[:, 2:] / 2), (target[:, :2] - target[:, 2:] / 2))# pred target都是[H,W,Y,X]# (Y,X)-(H,W) 左上角br = torch.min((pred[:, :2] + pred[:, 2:] / 2), (target[:, :2] + target[:, 2:] / 2))# (X,Y)+(H,W) 右下角area_p = torch.prod(pred[:, 2:], 1)  # HxWarea_g = torch.prod(target[:, 2:], 1)en = (tl < br).type(tl.type()).prod(dim=1)area_i = torch.prod(br - tl, 1) * enarea_u = area_p + area_g - area_iiou = (area_i) / (area_u + 1e-16)if self.loss_type == "iou":loss = 1 - iou ** 2elif self.loss_type == "giou":c_tl = torch.min((pred[:, :2] - pred[:, 2:] / 2), (target[:, :2] - target[:, 2:] / 2))c_br = torch.max((pred[:, :2] + pred[:, 2:] / 2), (target[:, :2] + target[:, 2:] / 2))area_c = torch.prod(c_br - c_tl, 1)giou = iou - (area_c - area_u) / area_c.clamp(1e-16)loss = 1 - giou.clamp(min=-1.0, max=1.0)# pred[:, :2]  pred[:, 2:]# (Y,X)        (H,W)# target[:, :2]  target[:, 2:]# (Y,X)        (H,W)elif self.loss_type == "diou":'''计算两个box的中心点距离d'''# d = math.sqrt((x2 - x1) ** 2 + (y2 - y1) ** 2)d = math.sqrt((pred[:, -1] - target[:, -1]) ** 2 + (pred[:, -2] - target[:, -2]) ** 2)# 左边xpred_l = pred[:, -1] - pred[:, -1] / 2target_l = target[:, -1] - target[:, -1] / 2# 上边ypred_t = pred[:, -2] - pred[:, -2] / 2target_t = target[:, -2] - target[:, -2] / 2# 右边xpred_r = pred[:, -1] + pred[:, -1] / 2target_r = target[:, -1] + target[:, -1] / 2# 下边ypred_b = pred[:, -2] + pred[:, -2] / 2target_b = target[:, -2] + target[:, -2] / 2'''计算两个box的bound的对角线距离'''bound_l = torch.min(pred_l, target_l)  # leftbound_r = torch.max(pred_r, target_r)  # rightbound_t = torch.min(pred_t, target_t)  # topbound_b = torch.max(pred_b, target_b)  # bottomc = math.sqrt((bound_r - bound_l) ** 2 + (bound_b - bound_t) ** 2)dloss = iou - (d ** 2) / (c ** 2)loss = 1 - dloss.clamp(min=-1.0, max=1.0)# Step1# def DIoU(a, b):# d = a.center_distance(b)# c = a.bound_diagonal_distance(b)# return IoU(a, b) - (d ** 2) / (c ** 2)# Step2-1# def center_distance(self, other):#    '''#    计算两个box的中心点距离#    '''#    return euclidean_distance(self.center, other.center)# Step2-2# def euclidean_distance(p1, p2):#    '''#    计算两个点的欧式距离#    '''#     x1, y1 = p1#    x2, y2 = p2#    return math.sqrt((x2 - x1) ** 2 + (y2 - y1) ** 2)# Step3# def bound_diagonal_distance(self, other):#    '''#    计算两个box的bound的对角线距离#    '''#    bound = self.boundof(other)#    return euclidean_distance((bound.x, bound.y), (bound.r, bound.b))# Step3-2# def boundof(self, other):#    '''#    计算box和other的边缘外包框,使得2个box都在框内的最小矩形#    '''#    xmin = min(self.x, other.x)#    ymin = min(self.y, other.y)#    xmax = max(self.r, other.r)#    ymax = max(self.b, other.b)#    return BBox(xmin, ymin, xmax, ymax)# Step3-3# def euclidean_distance(p1, p2):#    '''#    计算两个点的欧式距离#    '''#     x1, y1 = p1#    x2, y2 = p2#    return math.sqrt((x2 - x1) ** 2 + (y2 - y1) ** 2)if self.reduction == "mean":loss = loss.mean()elif self.reduction == "sum":loss = loss.sum()return loss

GitHub - Megvii-BaseDetection/YOLOX: YOLOX is a high-performance anchor-free YOLO, exceeding yolov3~v5 with MegEngine, ONNX, TensorRT, ncnn, and OpenVINO supported. Documentation: https://yolox.readthedocs.io/

这篇关于损失函数:DIOU loss手写实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/296226

相关文章

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

Python函数作用域与闭包举例深度解析

《Python函数作用域与闭包举例深度解析》Python函数的作用域规则和闭包是编程中的关键概念,它们决定了变量的访问和生命周期,:本文主要介绍Python函数作用域与闭包的相关资料,文中通过代码... 目录1. 基础作用域访问示例1:访问全局变量示例2:访问外层函数变量2. 闭包基础示例3:简单闭包示例4

Python实现字典转字符串的五种方法

《Python实现字典转字符串的五种方法》本文介绍了在Python中如何将字典数据结构转换为字符串格式的多种方法,首先可以通过内置的str()函数进行简单转换;其次利用ison.dumps()函数能够... 目录1、使用json模块的dumps方法:2、使用str方法:3、使用循环和字符串拼接:4、使用字符

Linux下利用select实现串口数据读取过程

《Linux下利用select实现串口数据读取过程》文章介绍Linux中使用select、poll或epoll实现串口数据读取,通过I/O多路复用机制在数据到达时触发读取,避免持续轮询,示例代码展示设... 目录示例代码(使用select实现)代码解释总结在 linux 系统里,我们可以借助 select、

Linux挂载linux/Windows共享目录实现方式

《Linux挂载linux/Windows共享目录实现方式》:本文主要介绍Linux挂载linux/Windows共享目录实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录文件共享协议linux环境作为服务端(NFS)在服务器端安装 NFS创建要共享的目录修改 NFS 配

通过React实现页面的无限滚动效果

《通过React实现页面的无限滚动效果》今天我们来聊聊无限滚动这个现代Web开发中不可或缺的技术,无论你是刷微博、逛知乎还是看脚本,无限滚动都已经渗透到我们日常的浏览体验中,那么,如何优雅地实现它呢?... 目录1. 早期的解决方案2. 交叉观察者:IntersectionObserver2.1 Inter