OpenCV实现二值图细化(骨架提取)

对二值图进行细化(骨架提取),也就是把每根线条细化到一个像素的宽度。有两个比较成熟的算法实现此功能,分别是Zhang-Suen算法和Guo-Hall算法。

我们下面使用OpenCVSharp,使用C#实现上述两个算法:

cs 复制代码
private static Mat ThinningIteration(Mat img, int iter, int thinningType)
{
    Mat marker = Mat.Zeros(img.Size(), MatType.CV_8UC1);
    int rows = img.Rows;
    int cols = img.Cols;
    marker.Col(0).SetTo(1);
    marker.Col(cols - 1).SetTo(1);
    marker.Row(0).SetTo(1);
    marker.Row(rows - 1).SetTo(1);

    //THINNING_ZHANGSUEN
    if (thinningType == 1)
    {
        marker.ForEachAsByte((byte* value, int* position) =>
        {
            int i = position[0];
            int j = position[1];
            if (i == 0 || j == 0 || i == rows - 1 || j == cols - 1)
                return;

            var ptr = (byte*)img.Ptr(i, j).ToPointer();

            // p9 p2 p3
            // p8 p1 p4
            // p7 p6 p5
            byte p2 = ptr[-cols];
            byte p3 = ptr[-cols + 1];
            byte p4 = ptr[1];
            byte p5 = ptr[cols + 1];
            byte p6 = ptr[cols];
            byte p7 = ptr[cols - 1];
            byte p8 = ptr[-1];
            byte p9 = ptr[-cols - 1];

            int neighbors = p9 | (p2 << 1) | (p3 << 2) | (p4 << 3) | (p5 << 4) | (p6 << 5) | (p7 << 6) | (p8 << 7);

            if (iter == 0)
                *value = lut_zhang_iter0[neighbors];
            else
                *value = lut_zhang_iter1[neighbors];
        });
    }
    //THINNING_GUOHALL
    else if (thinningType == 2)
    {
        marker.ForEachAsByte((byte* value, int* position) =>
        {
            int i = position[0];
            int j = position[1];
            if (i == 0 || j == 0 || i == rows - 1 || j == cols - 1)
                return;

            var ptr = (byte*)img.Ptr(i, j).ToPointer();

            // p9 p2 p3
            // p8 p1 p4
            // p7 p6 p5
            byte p2 = ptr[-cols];
            byte p3 = ptr[-cols + 1];
            byte p4 = ptr[1];
            byte p5 = ptr[cols + 1];
            byte p6 = ptr[cols];
            byte p7 = ptr[cols - 1];
            byte p8 = ptr[-1];
            byte p9 = ptr[-cols - 1];

            int neighbors = p9 | (p2 << 1) | (p3 << 2) | (p4 << 3) | (p5 << 4) | (p6 << 5) | (p7 << 6) | (p8 << 7);

            if (iter == 0)
                *value = lut_guo_iter0[neighbors];
            else
                *value = lut_guo_iter1[neighbors];
        });
    }

    img &= marker;
    return img;
}

public static void Thinning(InputArray input, OutputArray output, int thinningType)
{
    Mat processed = input.GetMat().Clone();
    processed /= 255;

    Mat prev = processed.Clone();
    Mat diff = new Mat();
    do
    {
        processed = ThinningIteration(processed, 0, thinningType);
        processed = ThinningIteration(processed, 1, thinningType);
        Cv2.Absdiff(processed, prev, diff);
        if (diff.CountNonZero() == 0) break;
        processed.CopyTo(prev);
    }
    while (true);

    processed *= 255;

    processed.CopyTo(output);
}

我们的测试框图如下图所示:

原二值图如下图所示:

细化(提取骨骼)之后的结果图如下:

Zhang-Suen算法

Guo-Hall算法

相关推荐
困死,根本不会1 小时前
OpenCV视觉舵机控制系统:从坐标检测到串口控制完整实现
人工智能·opencv·计算机视觉
Fleshy数模1 小时前
基于OpenCV实现指纹识别与验证:原理与实战
人工智能·opencv·计算机视觉
李白的粉3 小时前
基于ssm的校园宽带业务管理系统
java·毕业设计·ssm·课程设计·源代码·校园宽带业务管理系统
光羽隹衡3 小时前
计算机视觉——Opencv(摄像头实时风格迁移)
人工智能·opencv·计算机视觉
爱打代码的小林5 小时前
OpenCV 实战:基于 SIFT 特征匹配的图像认证系统
人工智能·opencv·计算机视觉
Daydream.V6 小时前
OpenCV高端操作——特征检测(附案例实战)
人工智能·opencv·计算机视觉
爱打代码的小林6 小时前
OpenCV 实战:为视频添加椒盐噪声并实现中值滤波去噪
人工智能·opencv·计算机视觉
纤纡.6 小时前
基于 OpenCV 的计算机视觉实战:从图像矫正到指纹识别
人工智能·opencv·计算机视觉
Westward-sun.6 小时前
OpenCV图像特征提取:Harris角点检测与SIFT特征提取实战
人工智能·opencv·计算机视觉
梦醒过后说珍重1 天前
医学图像超分辨率:如何构建“教科书级”的模型评测与交互式可视化流水线?
opencv