基于OpenCV实战:提取中心线
发布日期:2021-05-07 01:25:01 浏览次数:25 分类:原创文章

本文共 3524 字,大约阅读时间需要 11 分钟。

点击上方“小白学视觉”,选择加"星标"或“置顶

重磅干货,第一时间送达

本文转自|AI算法与图像处理

问题

前几天有个人问了我一个问题,问题是这样的,他有如下的一张二值图像:

怎么得到白色Blob中心线,他希望的效果如下:

显然OpenCV中常见的轮廓分析无法获得上面的中心红色线段,本质上这个问题是如何提取二值对象的骨架,提取骨架的方法在OpenCV的扩展模块中,另外skimage包也支持图像的骨架提取。这里就分别基于OpenCV扩展模块与skimage包来完成骨架提取,得到上述图示的中心线。

01

安装skimage与opencv扩展包

Python环境下安装skimage图像处理包与opencv计算机视觉包,只需要分别执行下面两行命令:

pip install opencv-contrib-pythonpip install skimage

导入使用

from skimage import morphology import cv2 as cv

02

使用skimage实现骨架提取

有两个相关的函数实现二值图像的骨架提取,一个是基于距离变换实现的medial_axis方法;另外一个是基于thin的skeletonize骨架提取方法。两个方法的代码实现分别如下:

 1def skeleton_demo(image): 2    gray = cv.cvtColor(image, cv.COLOR_BGR2GRAY) 3    ret, binary = cv.threshold(gray, 0, 255, cv.THRESH_BINARY | cv.THRESH_OTSU) 4    binary[binary == 255] = 1 5    skeleton0 = morphology.skeletonize(binary) 6    skeleton = skeleton0.astype(np.uint8) * 255 7    cv.imshow("skeleton", skeleton) 8    cv.waitKey(0) 9    cv.destroyAllWindows()101112def medial_axis_demo(image):13    gray = cv.cvtColor(image, cv.COLOR_BGR2GRAY)14    ret, binary = cv.threshold(gray, 0, 255, cv.THRESH_BINARY | cv.THRESH_OTSU)15    binary[binary == 255] = 116    skel, distance = morphology.medial_axis(binary, return_distance=True)17    dist_on_skel = distance * skel18    skel_img = dist_on_skel.astype(np.uint8)*25519    contours, hireachy = cv.findContours(skel_img, cv.RETR_EXTERNAL, cv.CHAIN_APPROX_SIMPLE)20    cv.drawContours(image, contours, -1, (0, 0, 255), 1, 8)2122    cv.imshow("result", image)23    cv.waitKey(0)24    cv.destroyAllWindows()

03

使用OpenCV实现骨架提取

OpenCV的图像细化的骨架提取方法在扩展模块中,因此需要直接安装opencv-python的扩展包。此外还可以通过形态学的膨胀与腐蚀来实现二值图像的骨架提取,下面的代码实现就是分别演示了基于OpenCV的两种骨架提取方法。代码分别如下:

 1def morph_find(image): 2    gray = cv.cvtColor(image, cv.COLOR_BGR2GRAY) 3    ret, binary = cv.threshold(gray, 0, 255, cv.THRESH_BINARY | cv.THRESH_OTSU) 4    kernel = cv.getStructuringElement(cv.MORPH_CROSS, (3, 3)) 5    finished = False 6    size = np.size(binary) 7    skeleton = np.zeros(binary.shape, np.uint8) 8    while (not finished): 9        eroded = cv.erode(binary, kernel)10        temp = cv.dilate(eroded, kernel)11        temp = cv.subtract(binary, temp)12        skeleton = cv.bitwise_or(skeleton, temp)13        binary = eroded.copy()1415        zeros = size - cv.countNonZero(binary)16        if zeros == size:17            finished = True1819    contours, hireachy = cv.findContours(skeleton, cv.RETR_EXTERNAL, cv.CHAIN_APPROX_SIMPLE)20    cv.drawContours(image, contours, -1, (0, 0, 255), 1, 8)21    cv.imshow("skeleton", image)22    cv.waitKey(0)23    cv.destroyAllWindows()242526def thin_demo(image):27    gray = cv.cvtColor(image, cv.COLOR_BGR2GRAY)28    ret, binary = cv.threshold(gray, 0, 255, cv.THRESH_BINARY | cv.THRESH_OTSU)29    thinned = cv.ximgproc.thinning(binary)30    contours, hireachy = cv.findContours(thinned, cv.RETR_EXTERNAL, cv.CHAIN_APPROX_SIMPLE)31    cv.drawContours(image, contours, -1, (0, 0, 255), 1, 8)32    cv.imshow("thin", image)33    cv.waitKey(0)34    cv.destroyAllWindows()

运行结果如下:

下载1:OpenCV-Contrib扩展模块中文版教程

在「小白学视觉」公众号后台回复:扩展模块中文教程即可下载全网第一份OpenCV扩展模块教程中文版,涵盖扩展模块安装、SFM算法、立体视觉、目标跟踪、生物视觉、超分辨率处理等二十多章内容。

下载2:Python视觉实战项目52讲

在「小白学视觉」公众号后台回复:Python视觉实战项目即可下载包括图像分割、口罩检测、车道线检测、车辆计数、添加眼线、车牌识别、字符识别、情绪检测、文本内容提取、面部识别等31个视觉实战项目,助力快速学校计算机视觉。

下载3:OpenCV实战项目20讲

在「小白学视觉」公众号后台回复:OpenCV实战项目20讲即可下载含有20个基于OpenCV实现20个实战项目,实现OpenCV学习进阶。

交流群

欢迎加入公众号读者群一起和同行交流,目前有SLAM、三维视觉、传感器、自动驾驶、计算摄影、检测、分割、识别、医学影像、GAN、算法竞赛等微信群(以后会逐渐细分),请扫描下面微信号加群,备注:”昵称+学校/公司+研究方向“,例如:”张三 + 上海交大 + 视觉SLAM“。请按照格式备注,否则不予通过。添加成功后会根据研究方向邀请进入相关微信群。请勿在群内发送广告,否则会请出群,谢谢理解~

上一篇:干货|python基础知识总结
下一篇:8个计算机视觉深度学习中常见的Bug

发表评论

最新留言

第一次来,支持一个
[***.219.124.196]2025年03月27日 02时46分30秒