图像轮廓提取

本文介绍了图像处理中的轮廓提取方法,包括掏空内部点法的Python实现和OpenCV库的轮廓检测功能。通过示例代码展示了如何使用cv2.findContours和cv2.drawContours函数进行轮廓检测和绘制,适用于二值图像的轮廓提取。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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. 源码仓库地址

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

### 使用MATLAB实现零件轮廓提取 为了有效提取机械零件的边缘轮廓,可以采用多种图像处理技术和特定的MATLAB函数组合。下面介绍一种综合的方法: #### 图像读取与预处理 首先加载待处理的灰度图像并对其进行必要的预处理操作,比如去噪和平滑化。 ```matlab gray_img = imread('part_image.png'); % 加载图片 figure; imshow(gray_img); title('原始图像'); blurred = imfilter(gray_img, fspecial('gaussian', [5 5], 2), 'replicate'); % 应用高斯滤波器平滑图像[^3] ``` #### 边缘检测 接着利用`edge()`函数执行边缘检测。此函数提供了几种不同的算子选项来寻找图像中的边界像素点。对于精度要求高的场合推荐选用Canny方法因为它能较好地区分真实和虚假边缘特征。 ```matlab BW1 = edge(blurred,'sobel'); % Sobel算子 BW2 = edge(blurred,'canny'); % Canny算子 figure; subplot(1,2,1); imshow(BW1);title('Sobel Edge Detection'); subplot(1,2,2); imshow(BW2);title('Canny Edge Detection'); ``` #### 后续增强与细化 有时还需要进一步加强或简化获得的结果,例如通过形态学运算去除细小结构或是填补断裂线条等。 ```matlab se = strel('disk',2); % 定义圆形结构元素 dilatedEdge = imdilate(BW2, se); % 膨胀操作连接断开部分 cleanedEdge = bwmorph(dilatedEdge,'thin',Inf); % 细化到单像素宽度 figure; imshow(cleanedEdge); title('最终清理后的边缘图'); ``` 这种方法不仅适用于一般情况下的物体轮廓捕捉,在工业自动化领域也有着广泛应用价值[^4]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值