C++ OpenCV自适应阈值Canny边缘检测

学更好的别人,

做更好的自己。

——《微卡智享》

29ea3856fb6d9d3b2897beb2805ba488.png

本文长度为1669,预计阅读5分钟

前言

Canny边缘检测速度很快,OpenCV中经常会用到Canny边缘检测,以前的Demo中使用Canny边缘检测都是自己手动修改高低阈值参数,最近正好要研究点小东西时,就想能不能做个自适应的阈值,在不影响整体效果的基础上不用手动调参,话不多说,直接开始。

f1bc38a36d4451359a0c3ed4f1a5c8d0.png

实现效果

794cd636129d43439b415bf7e6de9581.gif

b0322d6827a398599f069f207019f124.png

7ba4ef8b87ce876c8c9a8f64afbf559b.png

2f5694dfadeb5ad1d295faff149a0a96.png

6a5e7880fa7de52a5f7cfa7c03b40dc1.png

从上图中可以看出,命令行窗口中min和max就是求出的高低阈值,使用Canny边缘检测时直接就按这两个高低阈值处理的。

要实现自动阈值,方法就是求出图像的灰度直方图,直方图中找出中位数,然后根据中位数值设定一个标准差值,用中位数的值加上标准差来求出高低阈值。

 

#实现思路
1图像转为灰度图
2求出灰度直方图,并找到中位数
3根据中位数和设定的sigma值求出高低阈值
4使用Canny边缘检测

 

代码实现

cdc2a7ed26f9ce23fda40b509ce43e96.png

微卡智享

核心方法

根据上面的思路,最核心的方法就是求灰度图的中位数,求灰度图的直方图,然后根据直方图中出中位数,求直方图的原理可以看下面的动图。

865b68069b20d76dc08e3054cd5f29f0.gif

求中位数代码

//求Mat的中位数
int CvUtils::GetMatMidVal(Mat& img)
{
  //判断如果不是单通道直接返回128
  if (img.channels() > 1) return 128;
  int rows = img.rows;
  int cols = img.cols;
  //定义数组
  float mathists[256] = { 0 };
  //遍历计算0-255的个数
  for (int row = 0; row < rows; ++row) {
    for (int col = 0; col < cols; ++col) {
      int val = img.at<uchar>(row, col);
      mathists[val]++;
    }
  }


  int calcval = rows * cols / 2;
  int tmpsum = 0;
  for (int i = 0; i < 255; ++i) {
    tmpsum += mathists[i];
    if (tmpsum > calcval) {
      return i;
    }
  }
  return 0;
}

求出中位数后,我们再根据设置的sigma值求出高低阈值来。

50952c7126a1423ca44e164d83f823da.png

根据中位数求高低阈值代码

//求自适应阈值的最小和最大值
void CvUtils::GetMatMinMaxThreshold(Mat& img, int& minval, int& maxval, float sigma)
{
  int midval = GetMatMidVal(img);
  cout << "midval:" << midval << endl;
  // 计算低阈值
  minval = saturate_cast<uchar>((1.0 - sigma) * midval);
  //计算高阈值
  maxval = saturate_cast<uchar>((1.0 + sigma) * midval);
}


调用高低阈值Canny检测的代码

    //转换灰度图
    Mat gray, dst;
    cvtColor(src, gray, COLOR_BGR2GRAY);


    //高斯滤波
    GaussianBlur(gray, gray, Size(3, 3), 0.5, 0.5);


    //获取自适应阈值
    int minthreshold, maxthreshold;
    CvUtils::GetMatMinMaxThreshold(gray, minthreshold, maxthreshold);
    cout << "min:" << minthreshold << "  max:" << maxthreshold << endl;
    //Canny边缘提取
    Canny(gray, gray, minthreshold, maxthreshold);

这样自适应高低阈值的Canny边缘检测就完成了。

求中位数和取高低阈值的函数我放到了CvUtil的公共类中,等下个Demo完成后会调整一下位置,到时候一起上传上来。

9b7aeea215214ac642813106006c310a.png

扫描二维码

获取更多精彩

微卡智享

d2da5fe65d024ed15b117728693ebfb6.png

「 往期文章 」

C++ OpenCV4.5版本SIFT特征检测及匹配

趣玩算法--OpenCV华容道AI自动解题

整活!我是如何用OpenCV做了数字华容道游戏!(附源码)

 

  • 6
    点赞
  • 76
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论
OpenCV中的自适应阈值是一种基于图像局部区域的阈值处理方法。它可以自适应地调整每个像素的阈值,根据该像素周围的像素强度值来确定是否将其分为前景或背景。 在OpenCV中,可以使用`cv2.adaptiveThreshold()`函数来实现自适应阈值。该函数的语法如下: ``` cv2.adaptiveThreshold(src, maxValue, adaptiveMethod, thresholdType, blockSize, C[, dst]) ``` 其中,参数含义如下: - `src`:输入图像。 - `maxValue`:最大值(一般为255)。 - `adaptiveMethod`:自适应阈值算法类型,可选值为`cv2.ADAPTIVE_THRESH_MEAN_C`和`cv2.ADAPTIVE_THRESH_GAUSSIAN_C`,分别代表基于均值和高斯加权平均的算法。 - `thresholdType`:二值化类型,可选值为`cv2.THRESH_BINARY`和`cv2.THRESH_BINARY_INV`,分别代表二值化和反二值化。 - `blockSize`:计算阈值的像素邻域大小。 - `C`:阈值计算时的常数项,一般取0。 - `dst`(可选):输出图像。 下面是一个示例代码,演示如何使用自适应阈值对一张灰度图像进行二值化处理: ``` import cv2 img = cv2.imread('image.png', 0) # 使用自适应阈值进行二值化处理 thresh = cv2.adaptiveThreshold(img, 255, cv2.ADAPTIVE_THRESH_GAUSSIAN_C, cv2.THRESH_BINARY, 11, 2) cv2.imshow('Original Image', img) cv2.imshow('Adaptive Thresholding Result', thresh) cv2.waitKey(0) cv2.destroyAllWindows() ``` 在上述代码中,我们首先使用`cv2.imread()`函数读取一张灰度图像,然后使用`cv2.adaptiveThreshold()`函数对其进行自适应阈值二值化处理,最后使用`cv2.imshow()`函数将原始图像和处理结果显示出来。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Vaccae

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值