区域生长算法的一种C++实现
区域生长算法是一种图像分割方法,能够将图像中具有相同特征的连通区域分割出来,同时保证较好的边缘信息。
区域生长算法的优点是简单,容易实现;但空间和时间复杂度较高,对分割图像要求较高,否则容易形成孔洞和过分割。
区域生长算法的基本思想是首先获取分割区域的一个种子点,然后在种子点的周围搜索与该种子点有相似性质的像素点,合并到种子区域中。然后将合并的像素作为新的种子点继续搜索,直到种子区域中所有像素周围没有相似的像素点,算法结束。
如果要实现区域生长算法,基本算法流程是:
1. 选取种子点p(x0,y0),用堆栈表示种子区域,将种子点push到种子堆栈中
2. 将种子堆栈中第一个种子点pop出堆栈,并以该点为中心,遍历该中心8邻域像素
3. 判断遍历像素点是否已经在种子区域中,如果否,判断遍历像素点是否满足相邻种子点相似性,如果像素点(x,y)满足相似性,将(x,y)push到堆栈中
4. 重复步骤 2-3,直至种子堆栈为空。
从基本思想可以知道,影响区域生长算法的要素有三个:种子点的选取;搜索路径的选择;像素相似性的判断。
种子点的选取:一般情况下,区域生长算法是半交互式的分割算法,需要用户选取种子点。也可以是通过其他算法计算出来的种子点。
搜索路径的选择:搜索路径一般选择相邻的像素,以二维图像为例,一般为8邻域搜索,或者4邻域搜索;以三维图像为例,一般为26邻域搜索或者6邻域搜索。
像素相似性的判断:相似性一般以像素值的相近程度为判断标准,例如,可以设置一定灰度范围做为相似的标准。也可以通过计算满足某种形状或者性质作为判断标准。
接着根据上文中提到的算法,作者提出一种C++的实现方法,该实现不基于任何类库,以二维图像为例,假设图像的数据类型为char型。
首先,为了便于操作,需要定义种子点的类:
class Point2D
{
public:
Point2D(){}
Point2D(int ix, int iy)
{
this->x = ix;
this->y = iy;
} ~Point2D(){} Point2D operator+(const Point2D& a) const
{
return Point2D(x + a.x, y + a.y);
} Point2D operator-(const Point2D& a) const
{
return Point2D(x - a.x, y - a.y);
} bool operator=(const Point2D & a)
{
return (x == a.x && y == a.y);
} int x;
int y;
};
然后,定义种子点的邻域信息:
const Point2D PointShift2D[] =
{
Point2D(, ),
Point2D(, -),
Point2D(, -),
Point2D(-, -),
Point2D(-, ),
Point2D(-, ),
Point2D(, ),
Point2D(, )
};
然后,定义区域生长算法类的头文件:
class RegionGrowing
{
public:
RegionGrowing();
~RegionGrowing(); void SetInputData(char *pData, int width, int height); void SetSeedPoint(Point2D &p); void SetThreshold(int low, int hight); bool RegionGrow2D(); char* GetOutput(); private:
int LowThreshold;
int HighThreshold; int Width;
int Height; char *InputData;
char *OutputData; Point2D SeedPoint;
};
然后,是区域生长算法类的实现:
#include "RegionGrowing.h"
#include <stack> RegionGrowing::RegionGrowing()
{
this->InputData = nullptr;
this->OutputData = nullptr;
} RegionGrowing::~RegionGrowing()
{ } void RegionGrowing::SetInputData(char *pData, int width, int height)
{
this->InputData = pData;
this->Width = width;
this->Height = height;
} void RegionGrowing::SetSeedPoint(Point2D &p)
{
this->SeedPoint = p;
} void RegionGrowing::SetThreshold(int low, int high)
{
this->LowThreshold = low;
this->HighThreshold = high;
} bool RegionGrowing::RegionGrow2D()
{
if (this->InputData == nullptr || this->OutputData == nullptr)
{
return false;
} int index = this->SeedPoint.y * this->Width + this->SeedPoint.x;
int seedValue = this->InputData[index]; std::stack<Point2D> pointStack;
pointStack.push(this->SeedPoint); memset(this->OutputData, , sizeof(char)*this->Width*this->Height); while (!pointStack.empty())
{
Point2D topPoint = pointStack.top();
pointStack.pop(); for (int i = ; i < ; i++)
{
Point2D p = topPoint + PointShift2D[i]; index = p.y * this->Width + p.x; if (this->InputData[index] > this->LowThreshold &&
this->InputData[index] < this->HighThreshold &&
this->OutputData[index] == )
{
this->OutputData[index] = ;
pointStack.push(p);
}
}
} return true;
} char* RegionGrowing::GetOutput()
{
return this->OutputData;
}
声明本博客文章未特殊注明均为原创,转载请注明作者和原地址
区域生长算法的一种C++实现的更多相关文章
- 区域生长算法(附MATLAB代码实现)
一.理论概念 区域生长是按照事先定义的生长准则将一个像素或者子区域逐步聚合成一个完整独立的连通区域过程.对于图像感兴趣目标区域R,z为区域R上事先发现的种子点,按照规定的生长准则逐步将与种子点z一定邻 ...
- 区域生长算法 全局分类 C++ & matlab
// 注:本内容为作者原创,禁止在其他网站复述内容以及用于商业盈利,如需引用,请标明出处:https://www.cnblogs.com/lv-anchoret/ 今天我们来介绍用C++算法如何来实现 ...
- xgboost算法教程(两种使用方法)
标签: xgboost 作者:炼己者 ------ 欢迎大家访问我的简书以及我的博客 本博客所有内容以学习.研究和分享为主,如需转载,请联系本人,标明作者和出处,并且是非商业用途,谢谢! ------ ...
- Python数据结构与算法(几种排序)
数据结构与算法(Python) 冒泡排序 冒泡排序(英语:Bubble Sort)是一种简单的排序算法.它重复地遍历要排序的数列,一次比较两个元素,如果他们的顺序错误就把他们交换过来.遍历数列的工作是 ...
- [CC]区域生长算法——点云分割
基于CC写的插件,利用PCL中算法实现: void qLxPluginPCL::doRegionGrowing() { assert(m_app); if (!m_app) return; const ...
- 求质数算法的N种境界[1] - 试除法和初级筛法
★引子 前天,俺在<俺的招聘经验[4]:通过笔试答题能看出啥?>一文,以"求质数"作为例子,介绍了一些考察应聘者的经验.由于本文没有政治敏感内容,顺便就转贴到俺在CSD ...
- 【转】求质数算法的N种境界
原文地址:http://blog.csdn.net/program_think/article/details/7032600/ ★引子 前天,俺在<俺的招聘经验[4]:通过笔试答题能看出啥?& ...
- PHP 一致性哈希算法的一种简单实现
在分布式系统中,如果某业务可以由多个相同的节点处理,很容易想到用HASH的方式将业务请求分散到这些节点处理,比如memecache缓存等分 布式集群应用,如果只是简单的使用,不涉及用户用户状态等信息, ...
- 【转载】阻塞队列之三:SynchronousQueue同步队列 阻塞算法的3种实现
一.SynchronousQueue简介 Java 6的并发编程包中的SynchronousQueue是一个没有数据缓冲的BlockingQueue,生产者线程对其的插入操作put必须等待消费者的移除 ...
随机推荐
- Socket异步通信学习二
接下来是服务器部分,采用异步模式,新建了一个AsynServer类,用于存放socket服务器代码,主要有4个方法: 有一个全局socket,下面四个方法中都用到. Socket socket = n ...
- ifndef/define/endif 的作用
转载自百度百科 ,感谢度娘 1 2 3 #ifdef语句1 //程序2 #endif 可翻译为:如果宏定义了语句1则执行程序2. 作用:我们可以用它区隔一些与特定头文件.程序库和其他文件版本有关的代码 ...
- JAVA_HttpClientUtils
package org.mobiletrain.utils; import java.io.BufferedInputStream; import java.io.BufferedOutputStre ...
- 主流存储引擎详解:Innodb,Tokudb、Memory、MYISAM、Federated
主流存储引擎: Innodb:推荐使用,主力引擎,使用99%以上的场景 Tokudb:高速写入使用,日用量大量写入eg:500G可压缩为50G.适用于访问日志的写入,相对MYISAM有事务性,相对于I ...
- mysql数据库常用语句3
一:查询指定数据库中所有的表名 数据库名:test select table_name from information_schema.tables where table_schema='test' ...
- show variables 详解
back_log MySQL主线程检查连接并启动一个新线程这段时间内,可以设置多少个请求可以被存在堆栈中 connect_timeout 连接超时 检测方法nmap -p3306 数据库ip dela ...
- Navicate DataModel 注册码
注册信息: 姓 名:ttrar.com 组 织:(空) 序列号:NAVD-6CLM-6BKA-5TXK 内容来自: 可视化数据库设计工具(Navicat Data Modeler)1.0. ...
- Android——按钮的事件监听
关于Button按钮的四种事件监听方法总结 首先我们在activity_main.xml里面先定义一个Button空间 <RelativeLayout xmlns:android="h ...
- JAVA中REPLACE和REPLACEALL的区别(转)
replace和replaceAll是JAVA中常用的替换字符的方法,它们的区别是: 1)replace的参数是char和CharSequence,即可以支持字符的替换,也支持字符串的替换(Char ...
- Jquery 1.11.1 Checkbox checked 判断
if($(this).prop('checked'))