当前位置: 首页 > news >正文

opencv-python常用函数解析及参数介绍(八)——轮廓与轮廓特征

轮廓与轮廓特征

  • 前言
  • 1.获取轮廓
    • 通过膨胀与腐蚀获得轮廓
    • 通过梯度获取轮廓
    • 通过边缘检测获取轮廓
  • 2.寻找轮廓
    • 参数及作用对比
  • 3.轮廓特征


前言

在前面的文章中我们已经学会了使用膨胀与腐蚀、使用梯度、使用边缘检测的方式获得图像的轮廓,那么在获得轮廓后我们可以对图像进行什么样的操作呢?本文将介绍轮廓的绘制与轮廓特征的使用

1.获取轮廓

假设我们现在有这样一张名为feng.jpg的图片
在这里插入图片描述

同样为了表述方便,我们要先定义一个图像显示函数

def cv_show(img,name):
    cv2.imshow(name,img)
    cv2.waitKey()
    cv2.destroyAllWindows()

通过膨胀与腐蚀获得轮廓

在之前的博客:opencv-python常用函数解析及参数介绍(五)——腐蚀与膨胀中我们学到了三种获取轮廓的方式,大家可以查阅之前的博客获取细节,在本文中我们将使用膨胀减去腐蚀获取轮廓

img = cv2.imread('feng.jpg')
gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)
ret, thresh = cv2.threshold(gray, 230, 255, cv2.THRESH_BINARY)
kernel = np.ones((5,5),np.uint8) 
dilate = cv2.dilate(thresh, kernel, 1)
erosion = cv2.erode(thresh, kernel, 1)
contour_img = dilate-erosion # 获取轮廓
cv_show(contour_img, 'c')

在这里插入图片描述

通过梯度获取轮廓

本文使用效果最好的Sobel算子获取轮廓,其他的梯度计算方式请参考:opencv-python常用函数解析及参数介绍(六)——图像梯度

img = cv2.imread('feng.jpg')
gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)
ret, thresh = cv2.threshold(gray, 230, 255, cv2.THRESH_BINARY)

# 获取轮廓
contour_x = cv2.Sobel(thresh, cv2.CV_64F, 1, 0, ksize=3) 
contour_x = cv2.convertScaleAbs(contour_x)
contour_y = cv2.Sobel(thresh, cv2.CV_64F, 0, 1, ksize=3)
contour_y = cv2.convertScaleAbs(contour_y)
contour_img = cv2.addWeighted(contour_x, 0.5, contour_y, 0.5, 0)
cv_show(contour_img, 'c')

在这里插入图片描述

通过边缘检测获取轮廓

边缘检测的细节请参照:opencv-python常用函数解析及参数介绍(七)——边缘检测

img = cv2.imread('feng.jpg')
gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)
ret, thresh = cv2.threshold(gray, 230, 255, cv2.THRESH_BINARY)

# 获取轮廓
contour_img = cv2.Canny(img, 200, 250)
cv_show(contour_img, 'c')

在这里插入图片描述

2.寻找轮廓

下面我们使用效果最好的边缘检测得到的结果寻找轮廓

我们留意到,这个符号由多个图形组成,我们可以使用cv2.findContours函数找到每一部分的轮廓
findContours的参数为(图像,mode, method)

其中mode为轮廓检索模式,参数和作用如下

参数作用
RETR_EXTERNAL只检索最外面的轮廓;
RETR_LIST检索所有的轮廓,并将其保存到一条链表当中;
RETR_CCOMP检索所有的轮廓,并将他们组织为两层:顶层是各部分的外部边界,第二层是空洞的边界;
RETR_TREE检索所有的轮廓,并重构嵌套轮廓的整个层次

其中method为轮廓逼近方法,参数和作用如下

参数作用
CHAIN_APPROX_NONE以Freeman链码的方式输出轮廓,所有其他方法输出多边形(顶点的序列)。
CHAIN_APPROX_SIMPLE压缩水平的、垂直的和斜的部分,也就是,函数只保留他们的终点部分。

我们可以使用cv2.drawContours函数画出轮廓,其参数有(轮廓,轮廓索引,颜色模式,线条厚度)
轮廓索引是只第几个轮廓,当为-1时则画出所有轮廓,需要注意的是在绘制前需要先copy一份原图,否则原图也会被画上轮廓

参数及作用对比

这个过程只是为了方便展示效果,如果不理解不必强求

import math

for retr in ["cv2.RETR_EXTERNAL", "cv2.RETR_LIST", "cv2.RETR_CCOMP", "cv2.RETR_TREE"]:
    print(retr)
    contours, hierarchy = cv2.findContours(contour_img, eval(retr), cv2.CHAIN_APPROX_NONE)
    r_list = img

    row = int(math.sqrt(len(contours) + 2))
    col = math.ceil((len(contours) + 2) / row)+1

    plt.subplot(row,col, 1)
    plt.imshow(cv2.cvtColor(img, cv2.COLOR_BGR2RGB))
    plt.xticks([]), plt.yticks([])
    plt.title('origin')

    for i in range(len(contours)):
        res = cv2.drawContours(img.copy(), contours, i, (0, 0, 255), 2)
        plt.subplot(row, col, 2+i)
        plt.imshow(cv2.cvtColor(res, cv2.COLOR_BGR2RGB))
        plt.xticks([]), plt.yticks([])
        plt.title(str(i+1))

    plt.subplot(row, col, len(contours)+2)
    plt.imshow(cv2.cvtColor(cv2.drawContours(img.copy(), contours, -1, (0, 0, 255), 2), cv2.COLOR_BGR2RGB))
    plt.xticks([]), plt.yticks([])
    plt.title('all')
    plt.show()

在这里插入图片描述

可以看到除了第一个之外其他的效果相似,其实确实是这样的,其他的只不过是层次不一样,而第一个只取了外轮廓,所以看起来要比其他的轮廓少。
第二个参数只不过是存储方式不同,从描述上来看,一个适用于曲线较多的情况,另外一个适用于直线较多的情况。
同时我们还看到,这些图里面似乎有一些没什么变化的图,这是因为图中有噪点,所以有的点被误判成了轮廓,要去掉这种情况的点我们只需要求一下轮廓特征就好了,比如规定轮廓面积小于某个值就不算做轮廓或者周长小于某个值就不算做轮廓。

3.轮廓特征

在opencv中我们通过cv2.contourArea求面积,通过cv2.contourLength求周长,如果contoursLength第二个参数为True则求的周长为闭区间
我们来求一下第一个区间的面积和周长

contours, hierarchy=cv2.findContours(contour_img,cv2.RETR_TREE,cv2.CHAIN_APPROX_NONE)
res = cv2.drawContours(img.copy(), contours, 0, (0, 0, 255), 2)
cv_show(res,'res')
print(cv2.contourArea(contours[0]))
print(cv2.arcLength(contours[0], True))

在这里插入图片描述
在这里插入图片描述
下面看一下所有的面积和周长

contours, hierarchy=cv2.findContours(contour_img,cv2.RETR_TREE,cv2.CHAIN_APPROX_NONE)
for i in range(len(contours)):
    print(cv2.contourArea(contours[i]), cv2.arcLength(contours[i], True))

在这里插入图片描述
值为0的就是那些被误判成轮廓的点

相关文章:

  • flutter项目编译问题汇总
  • C++关联容器(复习题篇)
  • 02SpringCloudAlibaba服务注册中心—Eureka
  • opencv-python常用函数解析及参数介绍(七)——边缘检测
  • 14---实现文件上传和下载(头像上传功能)
  • Vue2学习笔记(四):计算属性(computed)和监事属性(watch)
  • 《信号与系统实验》实验 4:连续离散时间信号与系统的复频域分析实验
  • 【算法】kmp、Trie、并查集、堆
  • 2022年终总结与展望
  • (黑马C++)L06 重载与继承
  • Docker常用命令 - 黑马学习笔记
  • 抽象⼯⼚模式
  • 基于React Native开发的非法App破解记录
  • 年度征文 | 回顾2022,展望2023(我难忘的2022,我憧憬的2023)
  • JavaScript篇.day08-DOM,节点,事件,定时器,位置及坐标
  • 【刷算法】从上往下打印二叉树
  • 2017 年终总结 —— 在路上
  • extjs4学习之配置
  • Laravel 实践之路: 数据库迁移与数据填充
  • Laravel核心解读--Facades
  • Linux快速配置 VIM 实现语法高亮 补全 缩进等功能
  • node入门
  • python_bomb----数据类型总结
  • React-redux的原理以及使用
  • Redis中的lru算法实现
  • SpiderData 2019年2月16日 DApp数据排行榜
  • yii2中session跨域名的问题
  • 爱情 北京女病人
  • 猴子数据域名防封接口降低小说被封的风险
  • 记录:CentOS7.2配置LNMP环境记录
  • 两列自适应布局方案整理
  • 前端 CSS : 5# 纯 CSS 实现24小时超市
  • 区块链将重新定义世界
  • 让你成为前端,后端或全栈开发程序员的进阶指南,一门学到老的技术
  • 如何合理的规划jvm性能调优
  • 使用 @font-face
  • 首页查询功能的一次实现过程
  • 我是如何设计 Upload 上传组件的
  • 译米田引理
  • 白色的风信子
  • 阿里云API、SDK和CLI应用实践方案
  • 从如何停掉 Promise 链说起
  • ​中南建设2022年半年报“韧”字当头,经营性现金流持续为正​
  • (HAL)STM32F103C6T8——软件模拟I2C驱动0.96寸OLED屏幕
  • (ISPRS,2023)深度语义-视觉对齐用于zero-shot遥感图像场景分类
  • (超详细)2-YOLOV5改进-添加SimAM注意力机制
  • (附源码)springboot码头作业管理系统 毕业设计 341654
  • (附源码)ssm教师工作量核算统计系统 毕业设计 162307
  • (没学懂,待填坑)【动态规划】数位动态规划
  • (收藏)Git和Repo扫盲——如何取得Android源代码
  • (一)ClickHouse 中的 `MaterializedMySQL` 数据库引擎的使用方法、设置、特性和限制。
  • (一)为什么要选择C++
  • (转载)从 Java 代码到 Java 堆
  • (转载)在C#用WM_COPYDATA消息来实现两个进程之间传递数据
  • .NET Compact Framework 多线程环境下的UI异步刷新