区域生长算法的一种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必须等待消费者的移除 ...
随机推荐
- Android 自定义View修炼-高仿猎豹清理大师自定义内存开口圆环比例进度View
一.概述 看见猎豹清理大师的内存开口圆环比例进度 挺有意思的,于是就是想自己实现下这样的效果,于是反编译了猎豹清理 大师的app看了下,原来是有两张图,于是脑子里就过了下思路,利用上下两张图,旋转上面 ...
- cxf
一,cxf webService入门案例 1,jar包 注意版本 使用jdk6和apache-cxf-3.1.2,但cxf-3.1.2已经不支持jdk6,需要jdk7以上 版本用错会报java.lan ...
- Unicode 编码解码
1. Regex.Unescape(str);返回Unicode解码,非Unicode直接返回 /// <summary> /// 2.转为Unicode编码 /// ...
- linux云计算集群架构学习笔记:workstation 12.0 按装Red Hat Enterprise Linux 7(64位)
安装RHEL7.2 步骤: 1.安装虚拟机,按以下截图安装即可 步骤2: Ret hat 7.2 操作系统安装 rhel7因为许可报错解决
- Genymontion安装
Genymontion是什么 它提供的是Android的虚拟环境,和sdk的模拟器类似,能够支持windows,liunx,mac等操作系统,他比我们SDK的模拟器快很多,所以也是为什么很多开发人员会 ...
- mysql copy表或表数据常用的语句整理汇总
mysql copy表或表数据常用的语句整理汇总. 假如我们有以下这样一个表: id username password ----------------------------------- 1 a ...
- Spring分布式事务实现(适用于spring-tx 2.5)
http://log-cd.iteye.com/blog/807607 分布式事务是指操作多个数据库之间的事务,spring的org.springframework.transaction.jta.J ...
- sql的游标使用(转)
游标是邪恶的! 在关系数据库中,我们对于查询的思考是面向集合的.而游标打破了这一规则,游标使得我们思考方式变为逐行进行.对于类C的开发人员来着,这样的思考方式会更加舒服. 正常面向集合的思维方式是: ...
- Android笔记之adb命令应用实例1(手机端与PC端socket通讯下)
通过adb和Android通讯需要引用adb相关的组件到项目中,分别为:adb.exe,AdbWinApi.dll,AdbWinUsbApi.dll. 可以在XXX\sdk\platform-tool ...
- javascript form验证、完善 第24节
<html> <head> <title>Form对象</title> <style type="text/css"> ...