OpenCV——距离变换与分水岭算法的(图像分割)
C++: void distanceTransform(InputArray src, OutputArray dst, int distanceType, int maskSize)
参数详解:
InputArray src:输入的图像,一般为二值图像
OutputArray dst:输出的图像
int distanceType:所用的求解距离的类型、
It can be CV_DIST_L1, CV_DIST_L2 , or CV_DIST_C
mask_size 距离变换掩模的大小,可以是 3 或 5. 对 CV_DIST_L1 或 CV_DIST_C 的情况,参数值被强制设定为 3, 因为 3×3 mask 给出 5×5 mask 一样的结果,而且速度还更快。
- mask
- 用户自定义距离情况下的 mask。 在 3×3 mask 下它由两个数(水平/垂直位量,对角线位移量)组成, 5×5 mask 下由三个数组成(水平/垂直位移量,对角位移和 国际象棋里的马步(马走日))
- #include <opencv2/opencv.hpp>
- #include <iostream>
- using namespace cv;
- using namespace std;
- Mat src;
- int main(int argc, char** argv)
- {
- src = imread("分水岭.jpg");
- if (src.empty())
- {
- printf("Can not load Image...");
- return -;
- }
- imshow("input Image",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)[] = ;
- }
- }
- }
- imshow("black backgroung", src);
- //sharpen(提高对比度)
- Mat kernel = (Mat_<float>(, ) << , , , , -, , , , );
- //make it more sharp
- Mat imgLaplance;
- Mat sharpenImg = src;
- //拉普拉斯算子实现边缘提取
- filter2D(src, imgLaplance, CV_32F, kernel, Point(-, -), , BORDER_DEFAULT);//拉普拉斯有浮点数计算,位数要提高到32
- src.convertTo(sharpenImg, CV_32F);
- //原图减边缘(白色)实现边缘增强
- Mat resultImg = sharpenImg - imgLaplance;
- resultImg.convertTo(resultImg,CV_8UC3);
- imgLaplance.convertTo(imgLaplance, CV_8UC3);
- imshow("sharpen Image", resultImg);
- //转换成二值图
- Mat binary;
- cvtColor(resultImg, resultImg, CV_BGR2GRAY);
- threshold(resultImg, binary,,,THRESH_BINARY|THRESH_OTSU);
- imshow("binary image",binary);
- //距离变换
- Mat distImg;
- distanceTransform(binary,distImg,DIST_L1,,);
- normalize(distImg, distImg, , , NORM_MINMAX);
- imshow("dist image",distImg);
- //二值化
- threshold(distImg, distImg, 0.4, , THRESH_BINARY);
- imshow("dist binary image", distImg);
- //腐蚀(使得连在一起的部分分开)
- Mat k1 = Mat::ones(, , CV_8UC1);
- erode(distImg, distImg, k1);
- imshow("分开", distImg);
- //标记
- Mat dist_8u;
- distImg.convertTo(dist_8u,CV_8U);
- vector<vector<Point>> contours;
- findContours(dist_8u, contours, RETR_EXTERNAL, CHAIN_APPROX_SIMPLE, Point(, ));
- //创建标记
- Mat marker = Mat::zeros(src.size(),CV_32SC1);
- //画标记
- for (size_t i = ; i < contours.size(); i++)
- {
- drawContours(marker,contours,static_cast<int>(i),Scalar(static_cast<int>(i)+),-);
- }
- circle(marker, Point(, ), , Scalar(, , ), -);
- imshow("marker",marker*);
- //分水岭变换
- watershed(src,marker);//根据距离变换的标记,在原图上分离
- Mat water = Mat::zeros(marker.size(),CV_8UC1);
- marker.convertTo(water,CV_8UC1);
- bitwise_not(water, water,Mat());//取反操作
- //imshow("源 image", src);
- imshow("watershed Image", water);
- // generate random color
- 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));
- }
- // fill with color and display final result
- Mat dst = Mat::zeros(marker.size(), CV_8UC3);
- for (int row = ; row < marker.rows; row++) {
- for (int col = ; col < marker.cols; col++) {
- int index = marker.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::基于距离变换与分水岭的图像分割
什么是图像分割 图像分割(Image Segmentation)是图像处理最重要的处理手段之一 图像分割的目标是将图像中像素根据一定的规则分为若干(N)个cluster集合,每个集合包含一类像素. 根 ...
- Opencv距离变换distanceTransform应用——细化字符轮廓&&查找物体质心
Opencv中distanceTransform方法用于计算图像中每一个非零点距离离自己最近的零点的距离,distanceTransform的第二个Mat矩阵参数dst保存了每一个点与最近的零点的距离 ...
- matlab实现分水岭算法处理图像分割
此程序为优化后的分水岭算法,避免了图像过分割 I= imread('D:\Images\pic_loc\1870405130305041503.jpg'); imshow(I); h=fspecial ...
- OpenCV学习(23) 使用kmeans算法实现图像分割
本章我们用kmeans算法实现一个简单图像的分割.如下面的图像,我们知道图像分3个簇,背景.白色的任务,红色的丝带以及帽子. Mat img = cv::imread(&quo ...
- 基于标记的分水岭分割算法/OpenCV中距离变换
Opencv分水岭算法——watershed自动图像分割用法 OpenCV距离变换distanceTransform应用 图像分割作为图像识别的基础,在图像处理中占有重要地位,通常需要在进行图像分割算 ...
- python实现分水岭算法
目录: 问题:分水岭算法对图像分割很有作用,怎么把对象分割开来的?分水岭算法是比较完美的分割,跟前面的讲的轮廓不一样! (一)原理 (二)实现 (一)原理 opencv中的分水岭算法是基于距离变换的, ...
- python数字图像处理(19):骨架提取与分水岭算法
骨架提取与分水岭算法也属于形态学处理范畴,都放在morphology子模块内. 1.骨架提取 骨架提取,也叫二值图像细化.这种算法能将一个连通区域细化成一个像素的宽度,用于特征提取和目标拓扑表示. m ...
- OpenCV 学习笔记 04 深度估计与分割——GrabCut算法与分水岭算法
1 使用普通摄像头进行深度估计 1.1 深度估计原理 这里会用到几何学中的极几何(Epipolar Geometry),它属于立体视觉(stereo vision)几何学,立体视觉是计算机视觉的一个分 ...
随机推荐
- MySQL常用操作汇编
熟悉 我熟悉xxx,其实很多原来熟悉到能背的,如果长时间不用了几乎也就忘了.此时再说自己熟悉XXX就被认为是在吹牛B了,感觉不是很好.所谓温故而知新,对于天资不聪颖的,就是要在一遍一遍的复习实践中慢慢 ...
- Code Signal_练习题_chessBoardCellColor
Given two cells on the standard chess board, determine whether they have the same color or not. Exam ...
- python学习之老男孩python全栈第九期_day008作业
1. 文件a.txt内容:每一行内容分别为商品名字,价钱,个数,求出本次购物花费的总钱数apple 10 3tesla 100000 1mac 3000 2lenovo 30000 3chicken ...
- 3.配置Spring+SpringMvc+Mybatis(分库or读写分离)--Intellij IDAE 2016.3.5
作者QQ:1095737364 QQ群:123300273 欢迎加入! 建立好maven多模块项目后,开始使用ssm传统的框架:http://www.cnblogs.com/yysbol ...
- koajs项目之memcached实现session共享
在做nodejs服务的负载时要考虑到session共享的问题,一般常用的就是memcached方式实现的,本文主要介绍通过npm社区的几个模块轻松实现这个功能. 做koa的session一般会想到用k ...
- 1.String、StringBuffer与StringBuilder之间区别
1.三者在执行速度方面的比较:StringBuilder > StringBuffer > String 2.String <(StringBuffer,StringBuild ...
- 【读书笔记】iOS-网络-Cookie
Cookie是HTTP协议在首个版本之后加入的一个重要组件.它向服务器提供了追踪会话状态的能力,同时又无须维持客户端与服务器之间的连接.在浏览器客户端,Cookie值是由服务器通过请求提供的,,然后被 ...
- 3D-爱心
520把爱心送给她 用自己独有的方式表白,也是爱的一种体现! 所以呢,我就利用自己现有的知识,做了一个3D爱心! 今天是5月21日,博主在这里希望所有看到这个博客的朋友们能够拥有自己美好的爱情! 同时 ...
- 高德地图 JS API - 根据经纬度获取周边建筑地标
像我们经常用的微信或微博,发表动态时都有选择位置的功能,根据当前的定位获取附近的地标.利用高德地图我们就可以实现这样的功能. 1. 具体代码: // 高德地图查询周边 function aMapSea ...
- MySQL 性能监控4大指标——第二部分
[编者按]本文作者为 John Matson,主要介绍 mysql 性能监控应该关注的4大指标. 第一部分介绍了前两个指标:查询吞吐量与查询执行性能.本文将继续介绍另两个指标:MySQL 连接与缓冲池 ...