NMS(non maximum suppression)即非极大值抑制,广泛应用于传统的特征提取和深度学习的目标检测算法中。

NMS原理是通过筛选出局部极大值得到最优解。

在2维边缘提取中体现在提取边缘轮廓后将一些梯度方向变化率较小的点筛选掉,避免造成干扰。

在三维关键点检测中也起到重要作用,筛选掉特征中非局部极值。

在目标检测方面,无论是One-stage的SSD系列算法、YOLO系列算法还是Two-stage的基于RCNN系列的算法,非极大值抑制都是其中必不可少的一个组件,可以将较小分数的输出框过滤掉,同样,在三维基于点云的目标检测模型中亦有使用。

在现有的基于anchor的目标检测算法中,都会产生数量巨大的候选矩形框,这些矩形框有很多是指向同一目标,因此就存在大量冗余的候选矩形框。非极大值抑制算法的目的正在于此,它可以消除多余的框,找到最佳的物体检测位置。

IoU(Intersection over Union) :定位精度评价公式。

相当于两个区域交叉的部分除以两个区域的并集部分得出的结果。



IoU各个取值时的情况展示,一般来说,这个 Score > 0.5 就可以被认为一个不错的结果了。

IOU计算:

如何计算IoU(交并比)

选取两个矩形框左顶角的横,纵坐标的最大值,x21,y21;选取两个矩形框右下边角的横纵坐标的最小值,x12,y12;

  • 交集面积计算:
\[{Area(A \cap B)} = |x12 - x21| * |y12 - y21|
\]
  • 并集面积计算:
\[{Area(A \cup B)} = |x11 - x12| * |y11 - y12| + |x21 - x22| * |y21 - y22| - {Area(A \cap B)}
\]
  • 计算IOU公式
\[IoU = \frac {Area(A \cap B)} {Area(A \cup B)}
\]

算法流程如下:

  • 将所有框的得分排序,选中最高分及其对应的框
  • 遍历其余的框,如果和当前最高分框的重叠面积(IOU)大于一定阈值(常用的值为0.5左右),我们就将框删除。(为什么要删除,是因为超过设定阈值,认为两个框的里面的物体属于同一个类别,比如都属于狗这个类别。我们只需要留下一个类别的可能性框图即可。)
  • 从未处理的框中继续选一个得分最高的,重复上述过程。





代码如下:

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. """
  4. NMS function(Non-Maximum Suppression, 抑制不是极大值的元素)
  5. psedocode:
  6. 1. choose the highest score element a_1 in set B, add a_1 to the keep set C
  7. 2. compute the IOU between the chosen element(such as a_1) and others elements in set B
  8. 3. only keep the nums at set B whose IOU value is less than thresholds (can be set as >=0.5), delete the nums similiar
  9. to a_1(the higher IOU it is , the more interseciton between a_1 and it will have)
  10. 4. choose the highest score value a_2 left at set B and add a_2 to set C
  11. 5. repeat the 2-4 until there is nothing in set B, while set C is the NMS value set
  12. """
  13. import numpy as np
  14. # boxes表示人脸框的xywh4点坐标+相关置信度
  15. boxes = np.array([[100, 100, 210, 210, 0.72],
  16. [250, 250, 420, 420, 0.8],
  17. [220, 220, 320, 330, 0.92],
  18. [230, 240, 325, 330, 0.81],
  19. [220, 230, 315, 340, 0.9]])
  20. def py_cpu_nms(dets, thresh):
  21. # dets:(m,5) thresh:scaler
  22. x1 = dets[:, 0] # [100. 250. 220. 230. 220.]
  23. y1 = dets[:, 1] # [100. 250. 220. 240. 230.]
  24. x2 = dets[:, 2] # [210. 420. 320. 325. 315.]
  25. y2 = dets[:, 3] # [210. 420. 330. 330. 340.]
  26. areas = (y2 - y1 + 1) * (x2 - x1 + 1)
  27. scores = dets[:, 4] # [0 1 3 4 2]
  28. keep = []
  29. # index表示按照scores从高到底的相关box的序列号
  30. index = scores.argsort()[::-1] # [2 4 3 1 0]
  31. while index.size > 0:
  32. print("sorted index of boxes according to scores", index)
  33. # 选择得分最高的score直接加入keep列表中
  34. i = index[0]
  35. keep.append(i)
  36. # 计算score最高的box和其他box分别的相关交集坐标
  37. x11 = np.maximum(x1[i], x1[index[1:]]) # [220. 230. 250. 220.] 最高的被提走了,所以要从1开始取后 4位
  38. y11 = np.maximum(y1[i], y1[index[1:]]) # [230. 240. 250. 220.]
  39. x22 = np.minimum(x2[i], x2[index[1:]]) # [315. 320. 320. 210.]
  40. y22 = np.minimum(y2[i], y2[index[1:]]) # [330. 330. 330. 210.]
  41. print("x1 values by original order:", x1)
  42. print("x1 value by scores:", x1[index[:]]) # [220. 220. 230. 250. 100.]
  43. print("x11 value means replacing the less value compared" \
  44. " with the value by the largest score :", x11)
  45. # 计算交集面积
  46. w = np.maximum(0, x22 - x11 + 1) # the weights of overlap
  47. h = np.maximum(0, y22 - y11 + 1) # the height of overlap
  48. overlaps = w * h
  49. # 计算相关IOU值(交集面积/并集面积,表示边框重合程度,越大表示越相似,越该删除)
  50. # 重叠面积 /(面积1+面积2-重叠面积)
  51. ious = overlaps / (areas[i] + areas[index[1:]] - overlaps)
  52. # 只保留iou小于阈值的索引号,重复上步
  53. idx = np.where(ious <= thresh)[0]
  54. # 因为第一步index[0]已经被划走,所以需要原来的索引号需要多加一
  55. index = index[idx + 1]
  56. return keep
  57. import matplotlib.pyplot as plt
  58. def plot_bbox(ax, dets, c='b', title_name="title"):
  59. x1 = dets[:, 0]
  60. y1 = dets[:, 1]
  61. x2 = dets[:, 2]
  62. y2 = dets[:, 3]
  63. ax.plot([x1, x2], [y1, y1], c)
  64. ax.plot([x1, x1], [y1, y2], c)
  65. ax.plot([x1, x2], [y2, y2], c)
  66. ax.plot([x2, x2], [y1, y2], c)
  67. ax.set_title(title_name)
  68. if __name__ == '__main__':
  69. # 1.创建画板fig
  70. fig = plt.figure(figsize=(12, 6))
  71. # 参数解释,前两个参数 1,2 表示创建了一个一行两列的框 第三个参数表示当前所在的框
  72. ax1 = fig.add_subplot(1, 2, 1)
  73. ax2 = fig.add_subplot(1, 2, 2)
  74. plot_bbox(ax1, boxes, 'k', title_name="before nms") # before nms
  75. keep = py_cpu_nms(boxes, thresh=0.7)
  76. plot_bbox(ax2, boxes[keep], 'r', title_name="after nms") # after nms
  77. plt.show()

参考文献:

https://blog.csdn.net/weixin_42237113/article/details/105743296

https://blog.csdn.net/lz867422770/article/details/100019587

非极大值抑制(NMS)算法详解的更多相关文章

  1. 非极大值抑制(NMS)

    非极大值抑制顾名思义就是抑制不是极大值的元素,搜索局部的极大值.这个局部代表的是一个邻域,邻域有两个参数可变,一个是邻域的维数,二是邻域的大小.这里不讨论通用的NMS算法,而是用于在目标检测中提取分数 ...

  2. MATLAB的边缘检测函数中隐含的细化(非极大值抑制)算法

    前段时间做了一个车牌检测识别的项目,我的任务是将MATLAB中的算法移植成C++代码.在车牌区域提取的过程中,用到了水平方向的Sobel算子检测垂直边缘,一开始我直接把MATLAB中的 bw = ed ...

  3. pytorch实现yolov3(4) 非极大值抑制nms

    在上一篇里我们实现了forward函数.得到了prediction.此时预测出了特别多的box以及各种class probability,现在我们要从中过滤出我们最终的预测box. 理解了yolov3 ...

  4. 目标检测后处理之NMS(非极大值抑制算法)

    1.定义: 非极大值抑制算法NMS广泛应用于目标检测算法,其目的是为了消除多余的候选框,找到最佳的物体检测位置. 2.原理: 使用深度学习模型检测出的目标都有多个框,如下图,针对每一个被检测目标,为了 ...

  5. 非极大值抑制(Non-Maximum Suppression,NMS)

    概述 非极大值抑制(Non-Maximum Suppression,NMS),顾名思义就是抑制不是极大值的元素,可以理解为局部最大搜索.这个局部代表的是一个邻域,邻域有两个参数可变,一是邻域的维数,二 ...

  6. 非极大值抑制(NMS)

    转自:https://www.cnblogs.com/makefile/p/nms.html 概述 非极大值抑制(Non-Maximum Suppression,NMS),顾名思义就是抑制不是极大值的 ...

  7. 非极大值抑制Non-Maximum Suppression(NMS)

    非极大值抑制(Non-Maximum Suppression,NMS)   概述 非极大值抑制(Non-Maximum Suppression,NMS),顾名思义就是抑制不是极大值的元素,可以理解为局 ...

  8. IoU与非极大值抑制(NMS)的理解与实现

    1. IoU(区域交并比) 计算IoU的公式如下图,可以看到IoU是一个比值,即交并比. 在分子中,我们计算预测框和ground-truth之间的重叠区域: 分母是并集区域,或者更简单地说,是预测框和 ...

  9. Non-Maximum Suppression,NMS非极大值抑制

    Non-Maximum Suppression,NMS非极大值抑制概述非极大值抑制(Non-Maximum Suppression,NMS),顾名思义就是抑制不是极大值的元素,可以理解为局部最大搜索. ...

  10. 非极大值抑制算法(Python实现)

    date: 2017-07-21 16:48:02 非极大值抑制算法(Non-maximum suppression, NMS) 算法原理 非极大值抑制算法的本质是搜索局部极大值,抑制非极大值元素. ...

随机推荐

  1. Javaweb基础复习------Cookie+Session案例的实现(登录注册案例)

    Cookie对象的创建--Cookie cookie=new Cookie("key","value"); 发送Cookie:resp.addCookie(); ...

  2. 三天吃透Spring Cloud面试八股文

    本文已经收录到Github仓库,该仓库包含计算机基础.Java基础.多线程.JVM.数据库.Redis.Spring.Mybatis.SpringMVC.SpringBoot.分布式.微服务.设计模式 ...

  3. Vditor在原生JS中如何结合后端使用

    目录 1.Vditor介绍 2.如何在原生JS中结合后端使用 2.1 背景 2.2 正确使用方式 2.2.1 编辑页面 2.2.2 回显页面(修改页面) 2.2.3 预览页面 3.小结一下 1.Vdi ...

  4. 如何自动化测试你的接口?—— Rest Assured

    前言 不知道大家的项目是否都有对接口API进行自动化测试,反正像我们这种小公司是没有的.由于最近一直被吐槽项目质量糟糕,只能研发自己看看有什么接口测试方案.那么在本文中,我将探索如何使用 Rest A ...

  5. 五月二号java基础知识

    1.使用Runnable接口可以轻松实现多个线程共享相同数据,只要用用一个可运行对象作为参数创建多个线程就可以了2.当一个线程对共享的数据进行操作时,应使之成为一个"原子操作"即在 ...

  6. CommunityToolkit.Mvvm系列文章导航

    包 CommunityToolkit.Mvvm (又名 MVVM 工具包,以前名为 Microsoft.Toolkit.Mvvm) 是一个现代.快速且模块化的 MVVM 库. 它是 .NET 社区工具 ...

  7. v-if与v-for的优先级

    在Vue2中 v-for的优先级要高于v-if 在Vue3中 v-if 的优先级要高于v-for

  8. spring boot过滤器实现项目内接口过滤

    spring boot过滤器实现项目内接口过滤 业务 由于业务需求,存在两套项目,一套是路由中心,一套是业务系统. 现在存在问题是,路由中心集成了微信公众号与小程序模块功能,业务系统部署了多套服务. ...

  9. CF1738EBalance Addicts

    CF1738EBalance Addicts 原题: CF1738EBalance Addicts 目录 CF1738EBalance Addicts 题目大意 做法 思路 注意 code 题目大意 ...

  10. Godot 4.0 文件读取(C#)

    搞半天才弄明白Godot文件操作. Godot的文档总是试图让我使用自定义Resource来支持文件操作,但是我只需要读取纯文本. 读取纯文本 读取纯文本的方式如下: //Godot.FileAcce ...