• opencv--使用直方图找谷底进行确定分割阈值


    直方图原理就不说了,大家自行百度

    直方图可以帮助分析图像中的灰度变化,进而帮助确定最优二值化的灰度阈值(threshold level)。如果物体与背景的灰度值对比明显,此时灰度直方图就会包含双峰(bimodal histogram),即直方图中一般会有两个峰值,分别为图像的前景和背景。

    前景使得某个灰度区间的灰度值的数量急剧增加,就会产生一个峰值,同理背景会使另一个灰度区间的灰度值的数量急剧增加,就产生另外一个峰值,两峰间的谷底对应于物体边缘附近相对较少数目的像素点。

    这两个峰值之间的最小值一般就是最优二值化的分界点,通过这个分界点可以把前景和背景很好地分割开来。

    有时这两个峰值会有部分重叠,即左侧峰值的下降部分和右侧峰值的上升部分存在叠加。通常可以把自然界的信号看做高斯信号,即一个峰值对应一个高斯信号,当直方图中的两个高斯信号在某个灰度区域叠加的时候,其叠加区就形成了一个圆滑的谷底,就很难找到一个确切的位置(最优二值化的灰度值)把这两个峰值分开。

     

    1. float calculateThreshold(cv::Mat& img)
    2. {
    3. cv::Mat temp = img.clone();
    4. // 计算直方图
    5. cv::Mat hist;
    6. int histSize = 256; // 直方图尺寸
    7. float range[] = { 0, 256 }; // 像素值范围
    8. const float* ranges[] = { range };
    9. cv::calcHist(&img, 1, nullptr, cv::Mat(), hist, 1, &histSize, ranges);
    10. /*for (int i = 0; i < 21; i++)
    11. hist.at(i, 0) = 0.0;*/
    12. cv::normalize(hist, hist, 0, 1, cv::NORM_MINMAX);
    13. //hist.convertTo(hist, CV_32S);
    14. cv::GaussianBlur(hist, hist, cv::Size(0, 0),3,3);
    15. //cv::blur(hist, hist, cv::Size(1, 9),cv::Point(-1,-1));
    16. std::vector<float> peaks; // 存储峰值位置
    17. std::vector<float> valleys; // 存储低谷位置
    18. for (int i = 1; i < histSize - 1; i++) {
    19. //std::cout << std::fixed << std::setprecision(4);
    20. float currentValue = hist.at<float>(i);
    21. float prevValue = hist.at<float>(i - 1);
    22. float nextValue = hist.at<float>(i + 1);
    23. /*if (currentValue < 0.001)
    24. continue;*/
    25. // 具体情况需要修改currentValue>0.005的阈值
    26. if ((currentValue > prevValue && currentValue > nextValue && currentValue>0.005)) {
    27. std::cout << prevValue << " " << currentValue << " " << nextValue << std::endl;
    28. peaks.push_back(i); // 峰值
    29. }
    30. else if (currentValue < prevValue && currentValue < nextValue && currentValue>0.001) {
    31. std::cout << prevValue << " " << currentValue << " " << nextValue << std::endl;
    32. valleys.push_back(i); // 低谷
    33. }
    34. }
    35. if(valleys.size()>0)
    36. cv::threshold(temp, temp, valleys[0], 255, cv::THRESH_BINARY);
    37. // 创建直方图可视化图像
    38. int histWidth = 512;
    39. int histHeight = 400;
    40. cv::Mat histImage(histHeight, histWidth, CV_8UC3, cv::Scalar(0, 0, 0));
    41. cv::Mat hist_temp;
    42. // 归一化直方图数据
    43. cv::normalize(hist, hist_temp, 0, histImage.rows, cv::NORM_MINMAX, -1, cv::Mat());
    44. // 绘制直方图
    45. int binWidth = cvRound((double)histWidth / histSize);
    46. for (int i = 0; i < histSize; i++) {
    47. int binHeight = cvRound(hist_temp.at<float>(i));
    48. cv::line(histImage, cv::Point(i * binWidth, histHeight), cv::Point(i * binWidth, histHeight - binHeight), cv::Scalar(255, 255, 255));
    49. }
    50. if (valleys.size() > 0)
    51. return valleys[0];
    52. return 0;
    53. }

  • 相关阅读:
    直播美颜SDK对比评测:技术原理与应用实践
    设计模式学习笔记 - 开源实战四(下):总结Spring中用到的11种设计模式
    栈的应用场景(二)
    在线图片转文字怎么转?这种方法大家可以学会
    从微服务基本概念到核心组件-通过一个实例来讲解和分析
    【图像融合】基于matlab增强随机游走算法多焦点图像融合【含Matlab源码 1975期】
    07 | 自己动手,搭建HTTP实验环境
    css溢出隐藏的五种方法
    Rust的Slice切片
    C++----unordered_map unordered_set使用及模拟
  • 原文地址:https://blog.csdn.net/weixin_42398658/article/details/136362911