图像轮廓提取

1、轮廓提取

轮廓提取是提取出图像的外部轮廓特征,轮廓可能是边缘的一部分。

2、轮廓提取方法及Python实现

2.1 掏空内部点法

掏空内部点法的原理非常简单:如果原图中有一点为黑,且它的8个相邻点皆为黑色,则将该点删除,否则认为该点在图像的边缘,需要保留。依次处理图像中每一个像素,则最后留下来的就是图像的轮廓。对于非二值图像,需要先进行二值化处理。
代码如下:

def Get_contour(bin_img):
    contour_img = np.zeros(shape=(bin_img.shape),dtype=np.uint8)
    contour_img += 255
    h = bin_img.shape[0]
    w = bin_img.shape[1]
    for i in range(1,h-1):
        for j in range(1,w-1):
            if(bin_img[i][j]==0):
                contour_img[i][j] = 0
                sum = 0
                sum += bin_img[i - 1][j + 1]
                sum += bin_img[i][j + 1]
                sum += bin_img[i + 1][j + 1]
                sum += bin_img[i - 1][j]
                sum += bin_img[i + 1][j]
                sum += bin_img[i - 1][j - 1]
                sum += bin_img[i][j - 1]
                sum += bin_img[i + 1][j - 1]
                if sum ==  0:
                    contour_img[i][j] = 255

    return contour_img

效果如下(左侧是Otsu二值化图像;右侧是轮廓图像):
在这里插入图片描述

2.2 opencv-python中轮廓提取方法的应用

(1)opencv-python中使用cv2.findContours函数来检测图像的边缘,其函数原型如下:

contours, hierarchy = cv2.findContours(image, mode, method[, contours[, hierarchy[, offset]]])

参数说明如下:
image:输入图像;
mode:轮廓检索模式;
method:轮廓逼近方法;
contours:返回的轮廓;
hierachy:每条轮廓对应的属性;
offset:每个轮廓点移动的可选偏移量。
备注:image参数需要是二值图,而不是灰度图,返回结果是等高线和层次结构。

轮廓检索模式:
cv2.RETR_EXTERNAL:表示只检测外轮廓;
cv2.RETR_LIST:检测的轮廓,不建立等级关系;
cv2.RETR_CCOMP:建立两个等级的轮廓,上面的一层为外边界,里面的一层为内孔的边界信息。如果内孔内还有一个连通物体,这个物体的边界也在顶层;
cv2.RETR_TREE:建立一个等级树结构的轮廓。

轮廓逼近方法:
cv2.CHAIN_APPROX_NONE:存储所有的轮廓点,相邻的两个点的像素位置差不超过1,即 max(abs(x1-x2),abs(y2-y1))==1,一般不会用到;
cv2.CHAIN_APPROX_SIMPLE:压缩水平方向,垂直方向,对角线方向的元素,只保留该方向的终点坐标,例如一个矩形轮廓只需4个点来保存轮廓信息;
cv2.CHAIN_APPROX_TC89_L1,cv2.CV_CHAIN_APPROX_TC89_KCOS:使用teh-Chinl chain近似算法。

(2)轮廓发现之后,还要通过cv2.drawContours函数绘制轮廓,其函数原型如下:

image = cv2.drawContours(image, contours, contourIdx, color[, thickness[, lineType[, hierarchy[, maxLevel[, offset]]]]])

参数说明如下:
image:输入图像;
contours:轮廓,在Python中是一个list,就是cv2.findContours函数找出来的点集,一个列表;
contourIdx:轮廓的索引,指定绘制轮廓list中的哪条轮廓,要绘制所有轮廓,传递-1;
color:颜色;
thickness:厚度,如果是-1,表示填充;
lineType:线型;
hierarchy:层次结构的可选信息;
maxLevel:绘制轮廓的最大级别,0:仅绘制指定的轮廓,1:绘制轮廓和所有嵌套轮廓,2:绘制轮廓,所有嵌套轮廓,所有嵌套到嵌套的轮廓;
offset:轮廓偏移参数。

根据上面两个函数,测试代码如下:

# 第一步:读入图像
img = cv2.imread('lenna.jpg')

# 第二步:对图像做灰度处理
gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)

# 第三步:对图像做二值化处理
ret, thresh = cv2.threshold(gray, 127, 255, cv2.THRESH_BINARY)

# 第四步:获得图像的轮廓值
contours, heriachy = cv2.findContours(thresh, cv2.RETR_TREE, cv2.CHAIN_APPROX_NONE)

# 第五步:绘制图像轮廓
img = cv2.cvtColor(img,cv2.COLOR_BGR2RGB)
res = cv2.drawContours(img, contours, -1, (0, 0, 255), 1)

plt.imshow(res, cmap='gray')
plt.title('contour')
plt.axis('off')
plt.show()

效果如下:
在这里插入图片描述

3. 源码仓库地址

🌼图像处理、机器学习的常用算法汇总

  • 3
    点赞
  • 45
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: OpenCV提供了一个函数cv::findContours(),可以用于从二值图像中提取轮廓。 具体步骤如下: 1. 将彩色图像转化为灰度图像。 2. 对灰度图像进行二值化处理,得到二值图像。 3. 使用cv::findContours()函数提取轮廓。 4. 绘制轮廓。 下面是一个示例代码: ```python import cv2 import numpy as np # 读入图像 img = cv2.imread('image.jpg') # 转化为灰度图像 gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY) # 二值化处理 ret, binary = cv2.threshold(gray, 127, 255, cv2.THRESH_BINARY) # 提取轮廓 contours, hierarchy = cv2.findContours(binary, cv2.RETR_TREE, cv2.CHAIN_APPROX_SIMPLE) # 绘制轮廓 cv2.drawContours(img, contours, -1, (0, 0, 255), 2) # 显示图像 cv2.imshow("image", img) cv2.waitKey(0) cv2.destroyAllWindows() ``` 其中,cv2.findContours()函数的参数说明如下: - 第一个参数:二值图像。 - 第二个参数:轮廓检索模式。有以下四种模式: - cv2.RETR_EXTERNAL:只检测外轮廓。 - cv2.RETR_LIST:检测所有轮廓,不建立轮廓之间的层级关系。 - cv2.RETR_CCOMP:检测所有轮廓,并将轮廓分为两级,即外层轮廓和内层轮廓。 - cv2.RETR_TREE:检测所有轮廓,并建立轮廓之间的层级关系。 - 第三个参数:轮廓逼近方法。有以下三种方法: - cv2.CHAIN_APPROX_NONE:存储所有的轮廓点。 - cv2.CHAIN_APPROX_SIMPLE:压缩水平、垂直、对角线方向上的像素点,只保留端点。 - cv2.CHAIN_APPROX_TC89_L1、cv2.CHAIN_APPROX_TC89_KCOS:使用Teh-Chin链逼近算法。 返回值: - contours:检测到的轮廓,每个轮廓是一个Numpy数组。 - hierarchy:检测到的轮廓之间的层级关系,每个轮廓有四个值:[next, previous, child, parent]。其中,next表示下一个轮廓的索引,previous表示前一个轮廓的索引,child表示第一个子轮廓的索引,parent表示父轮廓的索引。如果当前轮廓没有子轮廓,则child=-1;如果当前轮廓没有父轮廓,则parent=-1。 ### 回答2: 在OpenCV中,图像轮廓提取是一种常用的图像处理技术,它可以帮助我们找到图像中对象的边界。轮廓提取可以用于许多应用,如形状识别、物体检测和图像分割等。 在OpenCV中,图像轮廓提取的主要步骤如下: 1. 将图像转换为灰度图像,这有助于减少噪声并简化处理。 2. 对图像进行阈值处理,将图像转换为二值图像。这样可以将对象与背景分离。 3. 对二值图像进行形态学操作,如腐蚀和膨胀,可以去除噪声并平滑边界。 4. 使用cv2.findContours()函数找到图像中的轮廓。该函数将返回一个包含所有轮廓的列表。 5. 可选地,可以使用cv2.drawContours()函数在原始图像上绘制轮廓。这可以帮助我们可视化轮廓提取的结果。 在进行图像轮廓提取时,还可以使用一些参数来调整轮廓提取的效果。例如,可以通过调整阈值值和形态学操作来控制轮廓的数量和精度。 总的来说,OpenCV提供了简单且强大的图像轮廓提取工具,它可以帮助我们提取图像中的对象边界,进而实现各种应用。学习和掌握轮廓提取技术对于图像处理计算机视觉相关领域的研究和应用非常重要。 ### 回答3: OpenCV是一个强大的计算机视觉库,具有丰富的图像处理功能,其中包括图像轮廓提取图像轮廓是指图像中连接相同颜色或强度的连续曲线的图像特征。在OpenCV中,图像轮廓提取是通过以下几个步骤实现的: 首先,我们需要将图像转换为灰度图像。这是因为灰度图像只有一个通道,更容易处理。可以使用cvtColor()函数将彩色图像转换为灰度图像。 然后,我们可以使用阈值函数将图像二值化。通过设定适当的阈值,我们可以将图像分为目标和背景两部分。可以使用threshold()函数设置阈值,并根据需要选择不同的阈值类型。 接下来,使用findContours()函数可以检测图像中的轮廓。该函数将按大小排序返回一系列轮廓,并存储为一系列点的集合。可以选择提取所有的轮廓,或者只提取特定大小的轮廓。还可以使用其他参数来调整轮廓的检测精度和准确性。 提取轮廓后,可以使用drawContours()函数将轮廓绘制在原始图像上。该函数可以选择绘制所有的轮廓或单独绘制其中的一个轮廓。 如果需要进一步处理轮廓,可以使用一些附加函数,如计算轮廓的面积、周长、边界框等。还可以对轮廓进行操作,例如填充、裁剪或平滑处理。 最后,通过使用imshow()函数可以将处理后的图像显示出来。 总之,OpenCV提供了一套完整且易于使用的函数,可以方便地从图像中提取轮廓。通过适当的处理和调整参数,我们可以根据实际需求获取准确的轮廓信息。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值