Python Opencv实现单目标检测的示例代码


Posted in Python onSeptember 08, 2020

一 简介

目标检测即为在图像中找到自己感兴趣的部分,将其分割出来进行下一步操作,可避免背景的干扰。以下介绍几种基于opencv的单目标检测算法,算法总体思想先尽量将目标区域的像素值全置为1,背景区域全置为0,然后通过其它方法找到目标的外接矩形并分割,在此选择一张前景和背景相差较大的图片作为示例。

Python Opencv实现单目标检测的示例代码

环境:python3.7 opencv4.4.0

二 背景前景分离

1 灰度+二值+形态学 轮廓特征和联通组件

根据图像前景和背景的差异进行二值化,例如有明显颜色差异的转换到HSV色彩空间进行分割。

1 原图

Python Opencv实现单目标检测的示例代码

2 灰度化

Python Opencv实现单目标检测的示例代码

3 二值化

Python Opencv实现单目标检测的示例代码

4 形态学处理

Python Opencv实现单目标检测的示例代码

5 提取轮廓并找出目标外接矩形

代码封装:

def get_roi_contours(image_path, morph_size, num_morph):
  '''
  参数详解:
  image_path:所需处理图片路径
  morph_size:形态学处理核的大小
  num_morph:进行形态学处理的次数
  '''
  image = cv2.imread(image_path, cv2.IMREAD_COLOR)
  #灰度转换
  gray_image = cv2.cvtColor(image, cv2.COLOR_BGR2GRAY)
  #二值化
  threhold, binary_image = cv2.threshold(gray_image, 0, 255, cv2.THRESH_BINARY_INV+cv2.THRESH_OTSU)
  #形态学操作
  kernel = cv2.getStructuringElement(shape=cv2.MORPH_RECT, ksize=morph_size)
  morph_image = cv2.morphologyEx(binary_image, cv2.MORPH_CLOSE, kernel)
  for i in range(num_morph-1):
    morph_image = cv2.morphologyEx(morph_image, cv2.MORPH_CLOSE, kernel)
  #查找轮廓
  contours, hierarchy = cv2.findContours(morph_image, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)
  #选取轮廓面积最大的轮廓
  area = 0
  max_area_index = 0
  for j in range(len(contours)):
    if area < cv2.contourArea(contours[j]):
      max_area_index = j
      area = cv2.contourArea(contours[j])
  rect = cv2.boundingRect(contours[max_area_index])
  return rect

6 通过联通组件找到外接矩形
代码封装:

def get_roi_ConCom(image_path, morph_size, num_morph):
  '''
  参数详解:
  image_path:所需处理图片路径
  morph_size:形态学处理核的大小
  num_morph:进行形态学处理的次数
  '''
  image = cv2.imread(image_path, cv2.IMREAD_COLOR)
  #灰度转换
  gray_image = cv2.cvtColor(image, cv2.COLOR_BGR2GRAY)
  #二值化
  threhold, binary_image = cv2.threshold(gray_image, 0, 255, cv2.THRESH_BINARY_INV+cv2.THRESH_OTSU)
  #形态学操作
  kernel = cv2.getStructuringElement(shape=cv2.MORPH_RECT, ksize=morph_size)
  morph_image = cv2.morphologyEx(binary_image, cv2.MORPH_CLOSE, kernel)
  for i in range(num_morph-1):
    morph_image = cv2.morphologyEx(morph_image, cv2.MORPH_CLOSE, kernel)
  #联通组件查询
  numlabels, components_img, stats, centers = cv2.connectedComponentsWithStats(morph_image, 8)
  #获取除背景外的所有联通组件
  stats_without_back = stats[1:]
  #获取除背景外的所有联通组件的面积最大值
  max_area = np.max(stats_without_back, axis=0)[-1]
  #获取面积最大联通组件的index
  max_area_index = stats_without_back[:, -1]==max_area
  rect = stats_without_back[max_area_index]
  return np.squeeze(rect)[0:4]

2 Kmeans聚类实现前景和背景的分离

1 kmeans聚类后的图像,由于簇的中心是随机初始化的,所以目标的像素值可能为0,也可能为1,若采用opencv的findContours则要求前景像素值为1。

Python Opencv实现单目标检测的示例代码

2 利用轮廓特征找外接矩形

由于Kmeans随机初始化簇中心导致前景目标像素不确定,采用边缘提取的方法再查找轮廓。

边缘图:

Python Opencv实现单目标检测的示例代码

代码封装:

def get_roi_Kmeans(image_path):
  image = cv2.imread(image_path, cv2.IMREAD_COLOR)
  image_data = image.reshape(-1, 3).astype(np.float32) #必须要转成浮点类型进行计算
  #簇内平方和,标签和每个簇的中心
  criteria = (cv2.TERM_CRITERIA_EPS + cv2.TERM_CRITERIA_COUNT, 10, 1)
  interia, label, centers = cv2.kmeans(image_data, 2, None, criteria, 5, cv2.KMEANS_RANDOM_CENTERS)
  #二值化,将标签为0的转换为255,即是目标
  label[label==0] = 255
  label[label==1] = 0
  #转换数据类型,轮廓查找要是uint8类型数据
  thresh_img = label.reshape(image.shape[0:2]).astype(np.uint8)
  x_grad = cv2.Sobel(thresh_img, cv2.CV_32F, 1, 0)
  y_grad = cv2.Sobel(thresh_img, cv2.CV_32F, 0, 1)

  x_grad = cv2.convertScaleAbs(x_grad) #ax + b 线性变换
  y_grad = cv2.convertScaleAbs(y_grad)
  
  dst = cv2.add(x_grad, y_grad, dtype=cv2.CV_16S) #将两种sobel的加起来就可以得到整个边缘
  dst = cv2.convertScaleAbs(dst)
  plt.imshow(dst, cmap='gray')
  #轮廓查找目标必须为1
  contours, hierarchy = cv2.findContours(dst, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)
  #获取外接矩形
  rect = cv2.boundingRect(contours[0])
  return rect

三 总结

单目标检测较为简单,只要合理利用目标和背景的差异便可将其分离出来。当然单目标检测的方法还有很多,比如有目标模板的时候可以采用模板匹配或者均值漂移,有足够的数据集时也可采用机器学习和深度学习方法。

到此这篇关于Python Opencv实现单目标检测的示例代码的文章就介绍到这了,更多相关Opencv 单目标检测内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Python 相关文章推荐
简单了解Python下用于监视文件系统的pyinotify包
Nov 13 Python
python编程使用selenium模拟登陆淘宝实例代码
Jan 25 Python
详解Python 协程的详细用法使用和例子
Jun 15 Python
python3将视频流保存为本地视频文件
Jun 20 Python
python微信好友数据分析详解
Nov 19 Python
python 实现视频流下载保存MP4的方法
Jan 09 Python
pycharm配置pyqt5-tools开发环境的方法步骤
Feb 11 Python
Python迭代器iterator生成器generator使用解析
Oct 24 Python
pandas 空数据处理方法详解
Nov 02 Python
Tensorflow累加的实现案例
Feb 05 Python
解决Python logging模块无法正常输出日志的问题
Feb 21 Python
python3 logging日志封装实例
Apr 08 Python
python获取本周、上周、本月、上月及本季的时间代码实例
Sep 08 #Python
Python 使用Opencv实现目标检测与识别的示例代码
Sep 08 #Python
Python requests接口测试实现代码
Sep 08 #Python
Python unittest装饰器实现原理及代码
Sep 08 #Python
Python selenium环境搭建实现过程解析
Sep 08 #Python
Python unittest生成测试报告过程解析
Sep 08 #Python
Python使用Selenium模拟浏览器自动操作功能
Sep 08 #Python
You might like
使用JavaScript创建新样式表和新样式规则
2016/06/14 PHP
php实现批量修改文件名称的方法
2016/07/23 PHP
PHP如何通过带尾指针的链表实现'队列'
2020/10/22 PHP
Javascript 跨域访问解决方案
2009/02/14 Javascript
9个JavaScript评级/投票插件
2010/01/18 Javascript
原生js实现改变随意改变div属性style的名称和值的结果
2013/09/26 Javascript
浅析javascript 定时器
2014/12/23 Javascript
详谈javascript中DOM的基本属性
2015/02/26 Javascript
简单讲解jQuery中的子元素过滤选择器
2016/04/18 Javascript
微信小程序 数据遍历的实现
2017/04/05 Javascript
Vue-cli创建项目从单页面到多页面的方法
2017/09/20 Javascript
vue 中滚动条始终定位在底部的方法
2018/09/03 Javascript
Koa 使用小技巧(小结)
2018/10/22 Javascript
如何对react hooks进行单元测试的方法
2019/08/14 Javascript
vue 点击其他区域关闭自定义div操作
2020/07/17 Javascript
Vant picker 多级联动操作
2020/11/02 Javascript
Python使用xlrd读取Excel格式文件的方法
2015/03/10 Python
给Python中的MySQLdb模块添加超时功能的教程
2015/05/05 Python
Python 2/3下处理cjk编码的zip文件的方法
2019/04/26 Python
python flask 如何修改默认端口号的方法步骤
2019/07/12 Python
详解用python生成随机数的几种方法
2019/08/04 Python
django基于存储在前端的token用户认证解析
2019/08/06 Python
python实现监控阿里云账户余额功能
2019/12/16 Python
python3连接MySQL8.0的两种方式
2020/02/17 Python
python安装和pycharm环境搭建设置方法
2020/05/27 Python
python的help函数如何使用
2020/06/11 Python
python 读取串口数据的示例
2020/11/09 Python
西班牙拥有最佳品牌的动物商店:Animalear.com
2018/01/05 全球购物
几个Linux面试题笔试题
2016/08/01 面试题
vue实现倒计时功能
2021/03/24 Vue.js
便利店促销方案
2014/02/20 职场文书
兴趣小组活动总结
2014/05/05 职场文书
党委班子纠正“四风”问题整改措施
2014/10/28 职场文书
军事理论课感想
2015/08/11 职场文书
幼儿园教师培训心得体会
2016/01/21 职场文书
导游词之西安骊山
2019/12/20 职场文书