opencv::基于距离变换与分水岭的图像分割
- 什么是图像分割
- 图像分割(Image Segmentation)是图像处理最重要的处理手段之一
- 图像分割的目标是将图像中像素根据一定的规则分为若干(N)个cluster集合,每个集合包含一类像素。
- 根据算法分为监督学习方法和无监督学习方法,图像分割的算法多数都是无监督学习方法 - KMeans
- 距离变换常见算法有两种
- - 不断膨胀/腐蚀得到
- - 基于倒角距离
- 分水岭变换常见的算法
- - 基于浸泡理论实现
- cv::distanceTransform(
- InputArray src,
- OutputArray dst,
- OutputArray labels, //离散维诺图输出
- int distanceType, // DIST_L1/DIST_L2,
- int maskSize, // 3x3,最新的支持5x5,推荐3x3、
- int labelType=DIST_LABEL_CCOMP //dst输出8位或者32位的浮点数,单一通道,大小与输入图像一致
- )
- cv::watershed(
- InputArray image,
- InputOutputArray markers
- )
- 处理流程
- . 将白色背景变成黑色-目的是为后面的变换做准备
- . 使用filter2D与拉普拉斯算子实现图像对比度提高,sharp
- . 转为二值图像通过threshold
- . 距离变换
- . 对距离变换结果进行归一化到[~]之间
- . 使用阈值,再次二值化,得到标记
- . 腐蚀得到每个Peak - erode
- . 发现轮廓 – findContours
- . 绘制轮廓- drawContours
- . 分水岭变换 watershed
- . 对每个分割区域着色输出结果
- int main(int argc, char** argv) {
- char input_win[] = "input image";
- char watershed_win[] = "watershed segmentation demo";
- Mat src = imread(STRPAHT2);
- if (src.empty()) {
- printf("could not load image...\n");
- return -;
- }
- namedWindow(input_win, CV_WINDOW_AUTOSIZE);
- imshow(input_win, src);
- // 将白色背景变成黑色-为后面的变换做准备
- for (int row = ; row < src.rows; row++) {
- for (int col = ; col < src.cols; col++) {
- if (src.at<Vec3b>(row, col) == Vec3b(, , )) {
- src.at<Vec3b>(row, col)[] = ;
- src.at<Vec3b>(row, col)[] = ;
- src.at<Vec3b>(row, col)[] = ;
- }
- }
- }
- //namedWindow("black background", CV_WINDOW_AUTOSIZE);
- //imshow("black background", src);
- // sharpen
- Mat kernel = (Mat_<float>(, ) << , , , , -, , , , );
- Mat imgLaplance;
- Mat sharpenImg = src;
- //使用filter2D与拉普拉斯算子实现图像对比度提高,sharp
- filter2D(src, imgLaplance, CV_32F, kernel, Point(-, -), , BORDER_DEFAULT);
- src.convertTo(sharpenImg, CV_32F);
- Mat resultImg = sharpenImg - imgLaplance;
- resultImg.convertTo(resultImg, CV_8UC3);
- imgLaplance.convertTo(imgLaplance, CV_8UC3);
- imshow("sharpen image", resultImg);
- // convert to binary
- Mat binaryImg;
- cvtColor(src, resultImg, CV_BGR2GRAY);
- // 转为二值图像通过threshold
- threshold(resultImg, binaryImg, , , THRESH_BINARY | THRESH_OTSU);
- imshow("binary image", binaryImg);
- Mat distImg;
- // 每一个非零点距离离自己最近的零点的距离
- distanceTransform(binaryImg, distImg, DIST_L1, CV_DIST_C, );
- // 归一化
- normalize(distImg, distImg, , , NORM_MINMAX);
- imshow("distance result", distImg);
- // 使用阈值,再次二值化,得到标记
- threshold(distImg, distImg, ., , THRESH_BINARY);
- Mat k1 = Mat::ones(, , CV_8UC1);
- // 膨胀/腐蚀
- erode(distImg, distImg, k1, Point(-, -));
- imshow("distance binary image", distImg);
- // markers
- Mat dist_8u;
- distImg.convertTo(dist_8u, CV_8U);
- vector<vector<Point>> contours;
- // 发现轮廓
- findContours(dist_8u, contours, RETR_EXTERNAL, CHAIN_APPROX_SIMPLE, Point(, ));
- // 绘制轮廓
- Mat markers = Mat::zeros(src.size(), CV_32SC1);
- for (size_t i = ; i < contours.size(); i++) {
- drawContours(markers, contours, static_cast<int>(i), Scalar::all(static_cast<int>(i) + ), -);
- }
- circle(markers, Point(, ), , Scalar(, , ), -);
- imshow("my markers", markers * );
- // 分水岭变换
- watershed(src, markers);
- Mat mark = Mat::zeros(markers.size(), CV_8UC1);
- markers.convertTo(mark, CV_8UC1);
- bitwise_not(mark, mark, Mat());
- imshow("watershed image", mark);
- // 对每个分割区域着色输出结果
- vector<Vec3b> colors;
- for (size_t i = ; i < contours.size(); i++) {
- int r = theRNG().uniform(, );
- int g = theRNG().uniform(, );
- int b = theRNG().uniform(, );
- colors.push_back(Vec3b((uchar)b, (uchar)g, (uchar)r));
- }
- Mat dst = Mat::zeros(markers.size(), CV_8UC3);
- for (int row = ; row < markers.rows; row++) {
- for (int col = ; col < markers.cols; col++) {
- int index = markers.at<int>(row, col);
- if (index > && index <= static_cast<int>(contours.size())) {
- dst.at<Vec3b>(row, col) = colors[index - ];
- }
- else {
- dst.at<Vec3b>(row, col) = Vec3b(, , );
- }
- }
- }
- imshow("Final Result", dst);
- waitKey();
- return ;
- }
opencv::基于距离变换与分水岭的图像分割的更多相关文章
- Blob分析--粘连颗粒检测 基于距离变换的分水岭区域分割 盆地与原连通域求交集
文章转自微信公众号:机器视觉那些事 *******************************************************************公众号:机器视觉那些事儿*** ...
- OpenCV——距离变换与分水岭算法的(图像分割)
C++: void distanceTransform(InputArray src, OutputArray dst, int distanceType, int maskSize) 参数详解: I ...
- [ZZ] 基于Matlab的标记分水岭分割算法
基于Matlab的标记分水岭分割算法 http://blog.sina.com.cn/s/blog_725866260100rz7x.html 1 综述 Separating touching obj ...
- 基于Matlab的标记分水岭分割算法
转自:http://blog.sina.com.cn/lyqmath 1 综述 Separating touching objects in an image is one of the more d ...
- Opencv距离变换distanceTransform应用——细化字符轮廓&&查找物体质心
Opencv中distanceTransform方法用于计算图像中每一个非零点距离离自己最近的零点的距离,distanceTransform的第二个Mat矩阵参数dst保存了每一个点与最近的零点的距离 ...
- opencv基于PCA降维算法的人脸识别
opencv基于PCA降维算法的人脸识别(att_faces) 一.数据提取与处理 # 导入所需模块 import matplotlib.pyplot as plt import numpy as n ...
- 距离变换DT
距离变换:计算区域中的每个点与最接近的区域外的点之间距离,把二值图象变换为灰度图象. 对于目标中一个点,距离变换的定义为改点与目标边界最近的距离. 目标点离边界约近则值越小,转换的点越暗:越远,值越大 ...
- SSE图像算法优化系列二十一:基于DCT变换图像去噪算法的进一步优化(100W像素30ms)。
在优化IPOL网站中基于DCT(离散余弦变换)的图像去噪算法(附源代码) 一文中,我们曾经优化过基于DCT变换的图像去噪算法,在那文所提供的Demo中,处理一副1000*1000左右的灰度噪音图像耗时 ...
- Win8Metro(C#)数字图像处理--2.25二值图像距离变换
原文:Win8Metro(C#)数字图像处理--2.25二值图像距离变换 [函数名称] 二值图像距离变换函数DistanceTransformProcess(WriteableBitmap sr ...
随机推荐
- 60 (OC)* 23中设计模式
git设计模式
- .netCore+Vue 搭建的简捷开发框架 (3)-- Services层实现
继续交作业: 上一篇作业中我们实现了 Repository仓储层的应用.并为我们的框架引入了EFCore 详见: .netCore+Vue 搭建的简捷开发框架 (2)--仓储层实现和EFCore 的使 ...
- [LeetCode] 由 “中缀表达式 --> 后缀表达式" 所想
如何利用栈解决问题. Ref: 如何在程序中将中缀表达式转换为后缀表达式? 本文的引申:如何手写语法分析器 实现调度场算法 “9+(3-1)*3+10/2” --> “9 3 1-3*+ 10 ...
- C++ 变量判定的螺旋法则
C++ 中一个标识符配合着各种修饰界定符,使得标识符的本意不那么直观一眼就能看出,甚至需要仔细分析,才能知道该标识符的具体你含义. 比如: void (*signal(int, void (*fp)( ...
- CDH6.3.0 - Cloudera Enterprise 6 Release Guide 安装准备篇
一.安装之前 Cloudera管理器的存储空间规划 ClouderaManager跟踪许多后台流程中的服务.作业和应用程序的指标.所有这些指标都需要存储.根据组织的大小,此存储可以是本地的或远程的,基 ...
- Java源码跟踪阅读技巧
转:https://www.jianshu.com/p/ab865109070c 本文基于Eclipse IDE 1.Quick Type Hierarchy 快速查看类继承体系. 快捷键:Ctrl ...
- 阿里云安装zk并连接javaAPI测试
1.安装 可参照Ubuntu 搭建Zookeeper服务进行安装并启动. 2.注意 阿里云环境开放2181端口 2.1 查看已开放端口: firewall-cmd --permanent --zone ...
- Drill 学习笔记之 入门体验
简介: Apache Drill是一个低延迟的分布式海量数据(涵盖结构化.半结构化以及嵌套数据)交互式查询引擎.分布式.无模式(schema-free) 是Google Dremel的开源实现,本质是 ...
- 关于ajax提交表单的一些实例及遇到的问题和解决办法
ajax提交的表单有两种情况: 第一种:input type类型没有file上传文件类型的表单 第二种:input type类型有file上传文件类型的表单 之所以分为两种:是因为原生ajax是不能提 ...
- JS基本数据类型和引用数据类型的区别及深浅拷贝
前言 首先我们先来了解一下什么叫栈堆,基本数据类型与引用数据类型 1.栈(stack)和堆(heap)stack为自动分配的内存空间,它由系统自动释放:而heap则是动态分配的内存,大小也不一定会自动 ...