1. 什么是图像分割
  2. 图像分割(Image Segmentation)是图像处理最重要的处理手段之一
  3. 图像分割的目标是将图像中像素根据一定的规则分为若干(N)个cluster集合,每个集合包含一类像素。
  4. 根据算法分为监督学习方法和无监督学习方法,图像分割的算法多数都是无监督学习方法 - KMeans
  5.  
  6. 距离变换常见算法有两种
  7. - 不断膨胀/腐蚀得到
  8. - 基于倒角距离
  9.  
  10. 分水岭变换常见的算法
  11. - 基于浸泡理论实现
  1. cv::distanceTransform(
  2. InputArray src,
  3. OutputArray dst,
  4. OutputArray labels, //离散维诺图输出
  5. int distanceType, // DIST_L1/DIST_L2,
  6. int maskSize, // 3x3,最新的支持5x5,推荐3x3、
  7. int labelType=DIST_LABEL_CCOMP //dst输出8位或者32位的浮点数,单一通道,大小与输入图像一致
  8. )
  9.  
  10. cv::watershed(
  11. InputArray image,
  12. InputOutputArray markers
  13. )
  1. 处理流程
  2. . 将白色背景变成黑色-目的是为后面的变换做准备
  3. . 使用filter2D与拉普拉斯算子实现图像对比度提高,sharp
  4. . 转为二值图像通过threshold
  5. . 距离变换
  6. . 对距离变换结果进行归一化到[~]之间
  7. . 使用阈值,再次二值化,得到标记
  8. . 腐蚀得到每个Peak - erode
  9. . 发现轮廓 findContours
  10. . 绘制轮廓- drawContours
  11. . 分水岭变换 watershed
  12. . 对每个分割区域着色输出结果
  1. int main(int argc, char** argv) {
  2. char input_win[] = "input image";
  3. char watershed_win[] = "watershed segmentation demo";
  4. Mat src = imread(STRPAHT2);
  5. if (src.empty()) {
  6. printf("could not load image...\n");
  7. return -;
  8. }
  9. namedWindow(input_win, CV_WINDOW_AUTOSIZE);
  10. imshow(input_win, src);
  11.  
  12. // 将白色背景变成黑色-为后面的变换做准备
  13. for (int row = ; row < src.rows; row++) {
  14. for (int col = ; col < src.cols; col++) {
  15. if (src.at<Vec3b>(row, col) == Vec3b(, , )) {
  16. src.at<Vec3b>(row, col)[] = ;
  17. src.at<Vec3b>(row, col)[] = ;
  18. src.at<Vec3b>(row, col)[] = ;
  19. }
  20. }
  21. }
  22. //namedWindow("black background", CV_WINDOW_AUTOSIZE);
  23. //imshow("black background", src);
  24.  
  25. // sharpen
  26. Mat kernel = (Mat_<float>(, ) << , , , , -, , , , );
  27. Mat imgLaplance;
  28. Mat sharpenImg = src;
  29. //使用filter2D与拉普拉斯算子实现图像对比度提高,sharp
  30. filter2D(src, imgLaplance, CV_32F, kernel, Point(-, -), , BORDER_DEFAULT);
  31. src.convertTo(sharpenImg, CV_32F);
  32. Mat resultImg = sharpenImg - imgLaplance;
  33.  
  34. resultImg.convertTo(resultImg, CV_8UC3);
  35. imgLaplance.convertTo(imgLaplance, CV_8UC3);
  36. imshow("sharpen image", resultImg);
  37.  
  38. // convert to binary
  39. Mat binaryImg;
  40. cvtColor(src, resultImg, CV_BGR2GRAY);
  41. // 转为二值图像通过threshold
  42. threshold(resultImg, binaryImg, , , THRESH_BINARY | THRESH_OTSU);
  43. imshow("binary image", binaryImg);
  44.  
  45. Mat distImg;
  46. // 每一个非零点距离离自己最近的零点的距离
  47. distanceTransform(binaryImg, distImg, DIST_L1, CV_DIST_C, );
  48.  
  49. // 归一化
  50. normalize(distImg, distImg, , , NORM_MINMAX);
  51. imshow("distance result", distImg);
  52.  
  53. // 使用阈值,再次二值化,得到标记
  54. threshold(distImg, distImg, ., , THRESH_BINARY);
  55. Mat k1 = Mat::ones(, , CV_8UC1);
  56. // 膨胀/腐蚀
  57. erode(distImg, distImg, k1, Point(-, -));
  58. imshow("distance binary image", distImg);
  59.  
  60. // markers
  61. Mat dist_8u;
  62. distImg.convertTo(dist_8u, CV_8U);
  63. vector<vector<Point>> contours;
  64. // 发现轮廓
  65. findContours(dist_8u, contours, RETR_EXTERNAL, CHAIN_APPROX_SIMPLE, Point(, ));
  66.  
  67. // 绘制轮廓
  68. Mat markers = Mat::zeros(src.size(), CV_32SC1);
  69. for (size_t i = ; i < contours.size(); i++) {
  70. drawContours(markers, contours, static_cast<int>(i), Scalar::all(static_cast<int>(i) + ), -);
  71. }
  72. circle(markers, Point(, ), , Scalar(, , ), -);
  73. imshow("my markers", markers * );
  74.  
  75. // 分水岭变换
  76. watershed(src, markers);
  77. Mat mark = Mat::zeros(markers.size(), CV_8UC1);
  78. markers.convertTo(mark, CV_8UC1);
  79. bitwise_not(mark, mark, Mat());
  80. imshow("watershed image", mark);
  81.  
  82. // 对每个分割区域着色输出结果
  83. vector<Vec3b> colors;
  84. for (size_t i = ; i < contours.size(); i++) {
  85. int r = theRNG().uniform(, );
  86. int g = theRNG().uniform(, );
  87. int b = theRNG().uniform(, );
  88. colors.push_back(Vec3b((uchar)b, (uchar)g, (uchar)r));
  89. }
  90.  
  91. Mat dst = Mat::zeros(markers.size(), CV_8UC3);
  92. for (int row = ; row < markers.rows; row++) {
  93. for (int col = ; col < markers.cols; col++) {
  94. int index = markers.at<int>(row, col);
  95. if (index > && index <= static_cast<int>(contours.size())) {
  96. dst.at<Vec3b>(row, col) = colors[index - ];
  97. }
  98. else {
  99. dst.at<Vec3b>(row, col) = Vec3b(, , );
  100. }
  101. }
  102. }
  103. imshow("Final Result", dst);
  104.  
  105. waitKey();
  106. return ;
  107. }

opencv::基于距离变换与分水岭的图像分割的更多相关文章

  1. Blob分析--粘连颗粒检测 基于距离变换的分水岭区域分割 盆地与原连通域求交集

    文章转自微信公众号:机器视觉那些事 *******************************************************************公众号:机器视觉那些事儿*** ...

  2. OpenCV——距离变换与分水岭算法的(图像分割)

    C++: void distanceTransform(InputArray src, OutputArray dst, int distanceType, int maskSize) 参数详解: I ...

  3. [ZZ] 基于Matlab的标记分水岭分割算法

    基于Matlab的标记分水岭分割算法 http://blog.sina.com.cn/s/blog_725866260100rz7x.html 1 综述 Separating touching obj ...

  4. 基于Matlab的标记分水岭分割算法

    转自:http://blog.sina.com.cn/lyqmath 1 综述 Separating touching objects in an image is one of the more d ...

  5. Opencv距离变换distanceTransform应用——细化字符轮廓&&查找物体质心

    Opencv中distanceTransform方法用于计算图像中每一个非零点距离离自己最近的零点的距离,distanceTransform的第二个Mat矩阵参数dst保存了每一个点与最近的零点的距离 ...

  6. opencv基于PCA降维算法的人脸识别

    opencv基于PCA降维算法的人脸识别(att_faces) 一.数据提取与处理 # 导入所需模块 import matplotlib.pyplot as plt import numpy as n ...

  7. 距离变换DT

    距离变换:计算区域中的每个点与最接近的区域外的点之间距离,把二值图象变换为灰度图象. 对于目标中一个点,距离变换的定义为改点与目标边界最近的距离. 目标点离边界约近则值越小,转换的点越暗:越远,值越大 ...

  8. SSE图像算法优化系列二十一:基于DCT变换图像去噪算法的进一步优化(100W像素30ms)。

    在优化IPOL网站中基于DCT(离散余弦变换)的图像去噪算法(附源代码) 一文中,我们曾经优化过基于DCT变换的图像去噪算法,在那文所提供的Demo中,处理一副1000*1000左右的灰度噪音图像耗时 ...

  9. Win8Metro(C#)数字图像处理--2.25二值图像距离变换

    原文:Win8Metro(C#)数字图像处理--2.25二值图像距离变换  [函数名称] 二值图像距离变换函数DistanceTransformProcess(WriteableBitmap sr ...

随机推荐

  1. 60 (OC)* 23中设计模式

    git设计模式

  2. .netCore+Vue 搭建的简捷开发框架 (3)-- Services层实现

    继续交作业: 上一篇作业中我们实现了 Repository仓储层的应用.并为我们的框架引入了EFCore 详见: .netCore+Vue 搭建的简捷开发框架 (2)--仓储层实现和EFCore 的使 ...

  3. [LeetCode] 由 “中缀表达式 --> 后缀表达式" 所想

    如何利用栈解决问题. Ref: 如何在程序中将中缀表达式转换为后缀表达式? 本文的引申:如何手写语法分析器 实现调度场算法 “9+(3-1)*3+10/2” --> “9 3 1-3*+ 10 ...

  4. C++ 变量判定的螺旋法则

    C++ 中一个标识符配合着各种修饰界定符,使得标识符的本意不那么直观一眼就能看出,甚至需要仔细分析,才能知道该标识符的具体你含义. 比如: void (*signal(int, void (*fp)( ...

  5. CDH6.3.0 - Cloudera Enterprise 6 Release Guide 安装准备篇

    一.安装之前 Cloudera管理器的存储空间规划 ClouderaManager跟踪许多后台流程中的服务.作业和应用程序的指标.所有这些指标都需要存储.根据组织的大小,此存储可以是本地的或远程的,基 ...

  6. Java源码跟踪阅读技巧

    转:https://www.jianshu.com/p/ab865109070c 本文基于Eclipse IDE 1.Quick Type Hierarchy 快速查看类继承体系. 快捷键:Ctrl ...

  7. 阿里云安装zk并连接javaAPI测试

    1.安装 可参照Ubuntu 搭建Zookeeper服务进行安装并启动. 2.注意 阿里云环境开放2181端口 2.1 查看已开放端口: firewall-cmd --permanent --zone ...

  8. Drill 学习笔记之 入门体验

    简介: Apache Drill是一个低延迟的分布式海量数据(涵盖结构化.半结构化以及嵌套数据)交互式查询引擎.分布式.无模式(schema-free) 是Google Dremel的开源实现,本质是 ...

  9. 关于ajax提交表单的一些实例及遇到的问题和解决办法

    ajax提交的表单有两种情况: 第一种:input type类型没有file上传文件类型的表单 第二种:input type类型有file上传文件类型的表单 之所以分为两种:是因为原生ajax是不能提 ...

  10. JS基本数据类型和引用数据类型的区别及深浅拷贝

    前言 首先我们先来了解一下什么叫栈堆,基本数据类型与引用数据类型 1.栈(stack)和堆(heap)stack为自动分配的内存空间,它由系统自动释放:而heap则是动态分配的内存,大小也不一定会自动 ...