当前位置: 首页 > 编程日记 > 正文

OpenCV 【四】————Watershed Algorithm(图像分割)——分水岭算法的原理及实现

分水岭算法实现(C++、opencv)

1.作用:

通常用于分割图像,主要实现以临近像素间的相似性作为重要的参考依据,从而将在空间位置上相近并且灰度值相近的像素点互相连接起来构成一个封闭的轮廓封闭性是分水岭算法的一个重要特征。 相对于基于阈值的图像分割,边缘检测等都不会考虑像素在空间关系上的相似性和封闭性这一概念,彼此像素间互相独立,没有统一性。分水岭算法较其他分割方法更具有思想性,更符合人眼对图像的印象。

2.实现:

#include <cmath>
#include <iostream>
#include <memory>
#include <opencv2/core/core.hpp>
#include <opencv2/highgui/highgui.hpp>
#include <opencv2/imgproc/imgproc.hpp>
#include <opencv2/video/background_segm.hpp>
#include "opencv2/calib3d/calib3d.hpp"
#include <opencv2/opencv.hpp>
#include <time.h>
#include <thread>
#include <future>
#include <chrono>
#include <vector>
#include "time.h"
#include <string>
#include <fstream>using namespace std;
using namespace cv;Vec3b RandomColor(int value)    //生成随机颜色函数
{value = value % 255;  //生成0~255的随机数RNG rng;int aa = rng.uniform(0, value);int bb = rng.uniform(0, value);int cc = rng.uniform(0, value);return Vec3b(aa, bb, cc);
}int main(int argc, char** argv) {cv::Mat rgb_image = cv::imread("../example/timg.jpg", CV_LOAD_IMAGE_UNCHANGED);cv::Mat rgb_image_blur;GaussianBlur(rgb_image, rgb_image_blur, Size(5, 5), 0, 0);cv::Mat rgb_image_canny;Canny(rgb_image_blur, rgb_image_canny, 10, 120, 3, false);cv::imshow("rgb_roi_binary", rgb_image_canny);vector<vector<Point>>contours;vector<Vec4i>hierarchy;findContours(rgb_image_canny, contours, hierarchy, RETR_LIST, CHAIN_APPROX_SIMPLE, Point());Mat imageContours = Mat::zeros(rgb_image.size(), CV_8UC1);  //轮廓	Mat marks(rgb_image.size(), CV_32S);   //Opencv分水岭第二个矩阵参数marks = Scalar::all(0);int index = 0;int compCount = 0;for (; index >= 0; index = hierarchy[index][0], compCount++){//对marks进行标记,对不同区域的轮廓进行编号,相当于设置注水点,有多少轮廓,就有多少注水点drawContours(marks, contours, index, Scalar::all(compCount + 1), 1, 8, hierarchy);drawContours(imageContours, contours, index, Scalar(255), 1, 8, hierarchy);}//我们来看一下传入的矩阵marks里是什么东西Mat marksShows;convertScaleAbs(marks, marksShows);imshow("marksShow", marksShows);imshow("轮廓", imageContours);cv::watershed(rgb_image, marks);Mat afterWatershed;convertScaleAbs(marks, afterWatershed);imshow("After Watershed", afterWatershed);//对每一个区域进行颜色填充Mat PerspectiveImage = Mat::zeros(rgb_image.size(), CV_8UC3);for (int i = 0; i < marks.rows; i++){for (int j = 0; j < marks.cols; j++){int index = marks.at<int>(i, j);if (marks.at<int>(i, j) == -1){PerspectiveImage.at<Vec3b>(i, j) = Vec3b(255, 255, 255);}else{PerspectiveImage.at<Vec3b>(i, j) = RandomColor(index);}}}imshow("After ColorFill", PerspectiveImage);//分割并填充颜色的结果跟原始图像融合Mat wshed;addWeighted(rgb_image, 0.4, PerspectiveImage, 0.6, 0, wshed);imshow("AddWeighted Image", wshed);cv::waitKey(0);return 0;
}

3.效果

4.函数原型

void watershed( InputArray image, InputOutputArray markers );

第一个参数 image,必须是一个8bit 3通道彩色图像矩阵序列。第二个参数 markers,根据Opencv官方文档主要可分为以下几步:

step 1:图像灰度化、滤波、Canny边缘检测、二值化等一系列戏台学操作,保证下一步输入可以正常传入findContours即可。

参数markers它应该包含不同区域的轮廓,每个轮廓有一个自己唯一的编号,轮廓的定位可以cv::findContours(输入时二值化图像或者灰度图)方法,这个是执行分水岭之前的要求。 

step2:查找轮廓,并且把轮廓信息按照不同的编号绘制到watershed的第二个入参merkers上,相当于标记注水点。

算法会根据markers传入的轮廓作为种子(注水点),对图像上其他的像素点根据分水岭算法规则进行判断,并对每个像素点的区域归属进行划定,直到处理完图像上所有像素点。而区域与区域之间的分界处的值被置为“-1”,以做区分。

step3:watershed分水岭运算,绘制分割出来的区域,视觉控还可以使用随机颜色填充,或者跟原始图像融合以下,以得到更好的显示效果。

就是说第二个入参markers必须包含了种子点信息。Opencv官方例程中使用鼠标划线标记,其实就是在定义种子,只不过需要手动操作,而使用findContours可以自动标记种子点。而分水岭方法完成之后并不会直接生成分割后的图像,还需要进一步的显示处理。

5.原理

       将上述三个步骤对应到三维空间中,findContours发现的轮廓值就是分水岭的“ling”,在三维空间中,在该处滴上一滴水,这滴水会滑向最小值平面,轮廓中间就是最小值点,也即注水点。

实际上, 在真实图像中,由于噪声点或者其它干扰因素的存在,使用分水岭算法常常存在过度分割的现象,这是因为很多很小的局部极值点的存在。为了解决过度分割的问题,可以使用基于标记(mark)图像的分水岭算法,就是通过先验知识,来指导分水岭算法,以便获得更好的图像分段效果。通常的mark图像,都是在某个区域定义了一些灰度层级,在这个区域的洪水淹没过程中,水平面都是从定义的高度开始的,这样可以避免一些很小的噪声极值区域的分割。

6.参考

【1】 https://docs.opencv.org/3.4.9/d7/d1b/group__imgproc__misc.html#ga3267243e4d3f95165d55a618c65ac6e1

【2】https://blog.csdn.net/dcrmg/article/details/52498440

【3】https://www.cnblogs.com/mikewolf2002/p/3304118.html

相关文章:

SQL脚本--有关压缩数据库日志

/*--压缩数据库的通用存储过程 压缩日志及数据库文件大小 因为要对数据库进行分离处理 所以存储过程不能创建在被压缩的数据库中 --邹建 2004.03(引用请保留此信息)--*/ /*--调用示例 exec p_compdb test--*/ use master --注意,此存储过程要建在master数据库中Go if exists …

【POJ】1308 Is It A Tree?((并查集 + set)or (map))

http://poj.org/problem?id1308 这个题数组开到200就可以了&#xff0c;但题目中貌似没有说呢&#xff1f; 读入每一对顶点&#xff0c;看看他们是否在同一个集合中&#xff0c;如果是的话&#xff0c;肯定成环&#xff0c;不是一棵树。 用set容器保存节点&#xff0c;最后…

Java程序员修炼之路(一)我们为什么选择Java

我们为什么选择Java大多数人选择Java可能只是因为听说Java前景好、Java比较好找工作、Java语言在TIOBE排行榜上一直位于前三等等之类的原因&#xff0c;但是Java具体好在哪里&#xff0c;心里却是没有什么概念的。其实我选择Java也是出于以上的原因&#xff0c;但是现在确实真正…

iOS10 权限崩溃问题

iOS10 权限崩溃问题 今天 手机升级了 iOS10然后用正在开发的项目 装了个ipa包&#xff0c;发现点击有关 权限访问 直接Crash了&#xff0c;并在控制台输出了一些信息&#xff1a; This app has crashed because it attempted to access privacy-sensitive data without a usage…

OpenCV 【六】————youtu(图像)——旋转保存图片

OpenCV Mat结构的图片 旋转顺时针90度 180度 270度 逆时针90度 Mat matRotateClockWise90(Mat src) {if (src.empty()){qDebug()<<"RorateMat src is empty!";}// 矩阵转置transpose(src, src);//0: 沿X轴翻转&#xff1b; >0: 沿Y轴翻转&#xff1b; <…

【HDU】2087 剪花布条 (KMP算法的应用)

可以参考&#xff1a;从头彻尾彻底理解KMP 可以用朴素的模式匹配算法&#xff0c;也可以使用KMP算法&#xff0c;KMP算法所用的时间较短 普通版 #include <iostream> #include <string>using namespace std;int main () {string s1,s2;while(cin >> s1){i…

081_Introducing trigger handler class

案例分析&#xff1a; 我们对一个Object写多个独立得Trigger。 但最终这不是最好的做法。 在Salesforce中&#xff0c;只为每个SObject设置一个触发器总是好的。 原因&#xff1a;每个独立触发器的执行顺序始终未定义。 因此&#xff0c;如果我们有多个触发器&#xff0c;它可能…

python-opencv 定位识别读表

import cv2 import numpy as np import math import matplotlib.pyplot as plt """ 函数的格式为&#xff1a;kmeans(data, K, bestLabels, criteria, attempts, flags) &#xff08;1&#xff09;data: 分类数据&#xff0c;最好是np.float32的数据&#xff0c…

使用 Vue 2.0 实现服务端渲染的 HackerNews

Vue 2.0 支持服务端渲染 (SSR)&#xff0c;并且是流式的&#xff0c;可以做组件级的缓存&#xff0c;这使得极速渲染成为可能。同时&#xff0c; 和 2.0 也都能够配合 SSR 提供同构路由和客户端 state hydration。vue-hackernews-2.0 是 Vue 作者在GitHub上面放的 Vue 2.0 的一…

【HDU】Flipper 3328 (stack + 模拟 + 英语阅读)

http://acm.hdu.edu.cn/showproblem.php?pid3328 stack模拟&#xff0c;做完这一题最大的感想就是学好英语真的很重要&#xff0c;看题看了半天&#xff0c;还理解错了&#xff0c;这一题&#xff0c;出现L的话&#xff0c;就把最左边的牌翻转&#xff0c;然后放到它旁边的那…

最新阿里Java技术面试题,看这一文就够了!

金三银四跳槽季即将到来&#xff0c;作为 Java 开发者你开始刷面试题了吗&#xff1f;别急&#xff0c;小编整理了阿里技术面试题&#xff0c;看这一文就够了&#xff01;阿里面试题目目录1&#xff1a;技术一面(基础面试题目)2&#xff1a;技术二面&#xff08;技术深度、技术…

OpenCV 【七】————边缘提取算子(图像边缘提取)——canny算法的原理及实现

canny边缘检测实现&#xff08;C、opencv&#xff09; 1.作用&#xff1a; 图像边缘信息主要集中在高频段&#xff0c;通常说图像锐化或检测边缘&#xff0c;实质就是高频滤波。我们知道微分运算是求信号的变化率&#xff0c;具有加强高频分量的作用。在空域运算中来说&#x…

【Python】序列解包 and * 和 ** 的区别

可以使用序列解包功能对多个变量进行赋值。 序列解包也可以用于列表和字典&#xff0c;但对字典使用时&#xff0c;默认是对字典的“键”进行操作&#xff1b; 如果需要对“键&#xff1a;值”进行操作&#xff0c;需要使用字典items()方法&#xff1b; 如果需要对字典的“值…

JavaScript学习笔记—— 4. 变量、作用域和内存问题

ECMAScript变量可能包含两种不同数据类型的值&#xff1a;基本类型值和引用类型值&#xff0c;其中基本类型值是简单的数据段&#xff0c;而引用类型值指的是那些可能由多个值构成的对象&#xff1b;对于5种基本类型数据&#xff1a;undefined&#xff0c; null&#xff0c;boo…

ApacheCN 学习资源汇总 2019.3

【主页】 apachecn.org 【Github】ApacheCN 暂时下线: 社区 暂时下线: cwiki 知识库 自媒体平台 微博&#xff1a;ApacheCN知乎&#xff1a;ApacheCNCSDN简书OSChina博客园We are ApacheCN Open Source Organization, not ASF! We are fans of AI, and have no relationship w…

python数据结构与算法:二叉树及三种遍历方式(先序遍历/中序遍历/后序遍历)

树的实现采用queue的形式&#xff1a; 树的三种遍历方式&#xff08;广度优先白能力法&#xff09;&#xff1a;先序遍历&#xff08;根左右&#xff09;&#xff0c;中序遍历&#xff08;左根右&#xff09;以及后序遍历&#xff08;左右根&#xff09; ###################…

【HDU】3635 Dragon Balls (带权并查集 一)

题目链接&#xff1a;http://acm.hdu.edu.cn/showproblem.php?pid3635 【问题描述】 有标号为1到n的n个龙珠&#xff0c;分别放在对应标号为1到n的n个城市里。 下面有两种操作: T A B表示把A龙珠所在城市的所有龙珠都转移到B龙珠所在的城市中 Q A 表示查询A&#xff0c;需…

php源码安全加密之PHP混淆算法.

php源码安全加密的前世今生,本想发在教程区中.不知道怎么发,就写在这里面吧.PHP加密,解密是一直的话题,本人菜鸟,今天就简单向大家介绍一下并说说其中原理.提供一些加密的混淆算法.一\PHP的加密总体上来说分以下2种:1\扩展组件类加密,代表有:zend\ionCube\SG\php_screw\bcompil…

Element 2.6.0 发布,基于 Vue 2.0 的桌面端组件库

开发四年只会写业务代码&#xff0c;分布式高并发都不会还做程序员&#xff1f; Element 2.6.0 发布了&#xff0c;Element 是一套为开发者、设计师和产品经理准备的基于 Vue 2.0 的桌面端组件库&#xff0c;提供了配套设计资源&#xff0c;帮助你的网站快速成型。由饿了么公…

【Python】随机函数

import random 1、random.random() 2、random.uniform(a,b) 3、random.randint(a,b) 4、random.randrange([start],stop[, step]) 5、random.choice(sequence) 6、random.shuffle(x[, random]) 7、random.sample(sequence,k) import random 1、random.random() 返回随…

IOS/Android模拟器运行APP调试方法

真机不是那么好获取的&#xff0c;模拟器用起来更方便 ios 安装ios-sim 安装 ios-sim&#xff0c;获取 APP 完整的product的文件包。 $ npm i -g ios-sim 启动app # ios-sim launch *.app - -devicetypeid 本地模拟器支持机型类型 id $ ios-sim launch **/BaiduBoxApp.app --d…

python数据结构与算法:单向链表

单链表&#xff1a;python实现及其对应的 增删查检 操作 ##################### P4.1-P4.8 单向链表 ########################### #coding:utf-8 class Node(object):def __init__(self,elem):self.elem elemself.next Noneclass SinglelinkList(object):""&quo…

Vue性能优化:如何实现延迟加载和代码拆分?

移动优先方法已经成为一种标准&#xff0c;但不确定的网络条件导致应用程序快速加载变得越来越困难。在本系列文章中&#xff0c;我将深入探讨我们在Storefront应用程序中所使用的Vue性能优化技术&#xff0c;你们也可以在自己的Vue应用程序中使用它们来实现快速加载。 Webpack…

GitHub怎样fork别人代码到自己仓库并进行贡献

在过程中可能遇到这个问题&#xff1a;https://www.cnblogs.com/q1104460935/p/8275833.html 这个博客应该可以解决 比如说现在有一个很牛逼的项目&#xff0c;我们进入项目地址&#xff0c; 想将这个项目复制到自己的github仓库&#xff0c;然后你还想将 仓库中的代码拉取到…

python数据结构与算法:单向循环列表

单向循环列表&#xff1a;python实现&#xff0c;及其对应的 增删查检 操作 ##################### P4.9-P4.12 循环链表 ########################### #coding:utf-8 class Node(object):def __init__(self,elem):self.elem elemself.next None class SinglecycleList(ob…

http权威指南-http连接管理

2019独角兽企业重金招聘Python工程师标准>>> HTTP连接管理 浏览器解析URL流程&#xff1a; 浏览器解析出域名&#xff1b;浏览器查询这个主机名的IP地址&#xff1b;浏览器获得端口号&#xff1b;浏览器发起到主机名IP地址端口的80连接&#xff1b;浏览器向服务器发…

在macos上基于python2.7安装PyQt5

在macos上基于python2.7安装PyQt5 在python3上面安装PyQt5是十分简单的&#xff0c;可是&#xff0c;在python2.7上安装这个东西&#xff0c;着实让人折腾了一把。要总结一下&#xff0c;年纪大了&#xff0c;记性不好。 首先要安装最新版的Qt和python2&#xff0c;命令如下&am…

python数据结构与算法:二分查找

二分查找&#xff1a;python 实现def binary_seaech(alist,item):"""二分查找 递归实现"""n len(alist)if n > 0:mid n // 2if alist[mid] item:return Trueelif item < alist[mid]:return binary_seaech(alist[:mid],item)else:retur…

使用maven镜像

综述 用maven做项目&#xff0c;最郁闷的莫过于某些依赖库下载不了。被墙了&#xff0c;你懂的。使用maven镜像仓库及其重要&#xff0c;特别是国内的镜像&#xff0c;可以有效缓解被墙疼痛。 常用的镜像 国外镜像 ibiblio.org <mirror> <id>ibiblio</id> &…

Jupyter Notebook 快捷键(基本)

Jupyter Notebook 快捷键 Jupyter Notebook 有两种键盘输入模式。编辑模式&#xff0c;允许你往单元中键入代码或文本&#xff1b;这时的单元框线是绿色的。命令模式&#xff0c;键盘输入运行程序命令&#xff1b;这时的单元框线是灰色。 命令模式 (按键 Esc 开启) Enter : …