一、前言

GDAL具有强大的图像读写功能,但是对常用图像处理算法的集成较少,OpenCV恰恰具有较强的图像处理能力,因此有效的结合两者对图像(遥感影像)的处理带来了极大的方便。那么如何实现GDAL与openCV间的数据交换成为影像处理中的关键步骤。接下来我将记录下:1 如何将GDAL读取的影像转化为openCV支持的的MAT格式?2 如何将处理后MAT数据转化为合适的图像格式存储?(PS:本人也是初次使用GDAL和openCV,代码很水。。。只是记录下自己学的,和大家交流下)

二、GDAL数据到openCV的MAT格式

关于GDAL数据到openCV的格式转化,网上已有部分资源,但是大多是针对单或者三通道的数据而言,对多通道图像(遥感的多光谱和高光谱影像)的转化不多,话不多说,先上代码:

 cv::Mat GDAL2Mat(const QString fileName)
{
GDALAllRegister(); // 注册。。。
GDALDataset *poDataset = (GDALDataset *)GDALOpen(fileName.toStdString().c_str(),GA_ReadOnly);
int tmpCols = poDataset->GetRasterXSize();
int tmpRows = poDataset->GetRasterYSize();
int tmpBandSize = poDataset->GetRasterCount();
double *tmpadfGeoTransform = new double[];
poDataset->GetGeoTransform(tmpadfGeoTransform); QVector <cv::Mat> imgMat; // 每个波段
float *pafScan = new float[tmpCols*tmpRows]; // 存储数据 for(int i = ;i< tmpBandSize;i++)
{
GDALRasterBand *pBand = poDataset->GetRasterBand(i+);
//pafScan = new float[tmpCols*tmpRows];
pBand->RasterIO(GF_Read,,,tmpCols,tmpRows,pafScan,
tmpCols,tmpRows,GDT_Float32,,);
cv::Mat tmpMat = cv::Mat(tmpRows,tmpCols,CV_32FC1,pafScan);
imgMat.push_back(tmpMat.clone());
}
delete []pafScan;
pafScan = NULL; cv::Mat img;
img.create(tmpRows,tmpCols,CV_32FC(tmpBandSize));
cv::merge(imgMat.toStdVector(),img);
imgMat.clear();
GDALClose((GDALDatasetH)poDataset);
return img;
}

思路就是:根据文件名获得其GDALDataset数据集,然后分波段(波段相当于通道)存储在格式为Vector<cv::Mat>的容器内,最后利用MAT的Merge函数,对通道数据进行组合。以上方法适合任意波段数据,对多通道影像,如遥感影像中多光谱和高光谱数据比较实用。但,存在一个问题:代码中红色部分,目的为释放poDataset的内存,但总会报错,注释后就没有问题了,不知道为啥,哪位大侠如果知道原因并且也恰巧路过此地,请给予帮助,谢谢!(问题解决了,GDALDataset数据集前不能释放其每个波段的指针,否则报错,代码已修改,下同)

三、MAT格式数据转化为GDAL数据集格式后并保存合适文件

思路是上面第二部分的逆过程。首先创建一个数据集和文件驱动,根据相关参数创建文件,并将多通道MAT数据通过CV::split函数进行通道分离,最后将通道数据与GDAL数据集的波段数据对应,一一写入数据集中。代码如下:

 bool Mat2File(cv::Mat img, const QString fileName)
{
if(img.empty()) // 判断是否为空
return ; const int nBandCount=img.channels();
const int nImgSizeX=img.cols;
const int nImgSizeY=img.rows; // 将通道分开
// imgMat每个通道数据连续
std::vector<cv::Mat> imgMat(nBandCount);
cv::split(img,imgMat); // 分波段写入文件
GDALAllRegister();
GDALDataset *poDataset; //GDAL数据集
GDALDriver *poDriver; //驱动,用于创建新的文件
poDriver = GetGDALDriverManager()->GetDriverByName("ENVI"); if(poDriver == NULL)
return ;
poDataset=poDriver->Create(fileName.toStdString().c_str(),nImgSizeX,nImgSizeY,nBandCount,
GDT_Float32,NULL);
// 循环写入文件
GDALRasterBand *pBand = NULL;
float *ppafScan = new float[nImgSizeX*nImgSizeY];
cv::Mat tmpMat;// = cv::Mat(nImgSizeY,nImgSizeX,CV_32FC1); int n1 = nImgSizeY;
int nc = nImgSizeX; for(int i = ;i<=nBandCount;i++)
{
pBand = poDataset->GetRasterBand(i);
tmpMat = imgMat.at(i-);
if(tmpMat.isContinuous())
{
nc = n1*nc;
n1 = ;
}
for(int r = ;r<n1;r++)
{
int tmpI = r*nImgSizeX;
float *p = tmpMat.ptr<float>(r);
for(int c = ;c<nc;c++)
{
ppafScan[tmpI+c] = p[c];
}
}
pBand->RasterIO(GF_Write,,,nImgSizeX,nImgSizeY,ppafScan,
nImgSizeX,nImgSizeY,GDT_Float32,,);
}
delete []ppafScan;
ppafScan = NULL;
GDALClose(poDataset);
return ;
}
 bool ChooseSample::Mat2File(std::vector<cv::Mat> imgMat, const QString fileName)
{
if(imgMat.empty()) // 判断是否为空
{
QMessageBox::information(this,"Message Error","Data NULL!");
return ;
} const int nBandCount=imgMat.size();
const int nImgSizeX=imgMat[].cols;
const int nImgSizeY=imgMat[].rows; // 分波段写入文件
GDALAllRegister();
GDALDataset *poDataset; //GDAL数据集
GDALDriver *poDriver; //驱动,用于创建新的文件
poDriver = GetGDALDriverManager()->GetDriverByName("ENVI"); if(poDriver == NULL)
return ;
poDataset=poDriver->Create(fileName.toStdString().c_str(),nImgSizeX,nImgSizeY,nBandCount,
GDT_Float32,NULL);
// 循环写入文件
GDALRasterBand *pBand = NULL;
float *ppafScan = new float[nImgSizeX*nImgSizeY];
cv::Mat tmpMat;// = cv::Mat(nImgSizeY,nImgSizeX,CV_32FC1); int n1 = nImgSizeY;
int nc = nImgSizeX; for(int i = ;i<=nBandCount;i++)
{
pBand = poDataset->GetRasterBand(i);
tmpMat = imgMat.at(i-);
if(tmpMat.isContinuous())
{
nc = n1*nc;
n1 = ;
}
for(int r = ;r<n1;r++)
{
int tmpI = r*nImgSizeX;
float *p = tmpMat.ptr<float>(r);
for(int c = ;c<nc;c++)
{
ppafScan[tmpI+c] = p[c];
}
}
pBand->RasterIO(GF_Write,,,nImgSizeX,nImgSizeY,ppafScan,
nImgSizeX,nImgSizeY,GDT_Float32,,);
}
delete []ppafScan;
ppafScan = NULL;
GDALClose(poDataset);
return ;
}

同样有如上的困扰,每当释放内存就会报错(代码中红色字体处)。此外,关于cv::split函数有一个小的细节问题,如下:

     //    将通道分开
// imgMat每个通道数据连续
std::vector<cv::Mat> imgMat(nBandCount);
cv::split(img,imgMat); // imgMat每个通道数据不连续
QVector<cv::Mat> imgMat(nBandCount);
cv::split(img,imgMat.toStdVector());

1. GDAL与OpenCV2.X数据转换(适合多光谱和高光谱等多通道的遥感影像)的更多相关文章

  1. GDAL与OpenCV2.X数据转换(适合多光谱和高光谱等多通道的遥感影像)

    一.前言 GDAL具有强大的图像读写功能,但是对常用图像处理算法的集成较少,OpenCV恰恰具有较强的图像处理能力,因此有效的结合两者对图像(遥感影像)的处理带来了极大的方便.那么如何实现GDAL与o ...

  2. 基于GDAL的遥感影像显示(C#版)

    基于GDAL的遥感影像显示(C#版) - 菜菜的专栏 - 博客频道 - CSDN.NET  http://blog.csdn.net/RSyaoxin/article/details/9220735

  3. 【遥感影像】Python GDAL 像素与坐标对应

    转:https://blog.csdn.net/theonegis/article/details/50805520 https://blog.csdn.net/wsp_1138886114/arti ...

  4. gdal 遥感影像水体数据提取

  5. GDAL多光谱与全色图像融合简单使用

    目录 简述 C++代码 效果对比 GDAL融合效果和原始多光谱波段对比 GDAL融合效果和原始全色波段对比 ARCGIS融合效果与原始全色和多光谱对比 GDAL融合效果与ArcGIS融合效果对比 简述 ...

  6. AE + GDAL实现影像按标准图幅分割(上)

    最近有个项目,其中有个功能是要将遥感影像按标准图幅分割,一开始用AE的接口,慢的让人抓狂,就改用GDAL,速度提升很大.我主要通过http://blog.csdn.net/liminlu0314/学习 ...

  7. 使用GDAL进行RPC坐标转换

    使用GDAL进行RPC坐标转换 对于高分辨率遥感卫星数据而言,目前几乎都提供了有理函数模型(RFM)来进行图像校正(SPOT系列提供了有理函数模型之外还提供了严格轨道模型).对遥感影像进行校正目前最常 ...

  8. GDAL指定自定义的金字塔目录

    缘起 对于一般的遥感影像文件,金字塔文件默认都是与影像文件放在同一个目录下,金字塔文件名一般与源影像文件名相同,但后缀名不同.或者金字塔内建于影像内部,但这不是这里所涉及的. 在使用ArcGIS桌面版 ...

  9. 基于GDAL的栅格图像空间插值预处理

    转自 基于GDAL的栅格图像空间插值预处理——C语言版 基于GDAL的栅格图像预处理 前言 栅格数据和矢量数据构成空间数据的主要来源,怎样以开源方式读取并处理这些空间数据?目前有多种开源支持包,这里只 ...

随机推荐

  1. MongoDB 和 mySql 的关系

    1. mysql 和 MongoDb MySQL与MongoDB都是开源的常用数据库,但是MySQL是传统的关系型数据库,MongoDB则是非关系型数据库,也叫文档型数据库,是一种NoSQL的数据库. ...

  2. CodeSmith使用总结--读取一个表试试

    我感觉CodeSmith对于我的最大用途是不用我手动创建每个表的Model.BLL和DAL了,那些繁琐的工作真的让我很无语. CodeSmith要读取数据库中的表就要先连接数据库. 新建一个数据库连接 ...

  3. Javascript基础引用类型之Object

    虽然说ECMAScript也是一门对象语言,但是它和其他面向对象语言还是有区别的,它不具有类和接口等基本结构.所以在ECMAScript中一般说类指的是引用类型.创建Object实例的方式有两种: 第 ...

  4. Socket学习笔记

    ..........(此处略去万万字)学习中曲折的过程不介绍了,直接说结果 我的学习方法,问自己三个问题,学习过程将围绕这三个问题进行 what:socket是什么 why:为什么要使用socket ...

  5. Eclipse搭建Android开发环境(安装ADT,Android4.4.2)(转)

    使用Eclipse做Android开发,需要先在Eclipse上安装ADT(Android Development Tools)插件. 1.安装JDK 1.7 JDK官网http://www.orac ...

  6. Java ----------- SQL语句总结(更新中。。。。。。)

    #对数据库的操作 *创建数据库 CREATE DATABASE database_name:database_name为创建的数据库的变量名称. #对表的操作

  7. oracle学习笔记(二)表的查询

    --oracle表的管理 --创建表 )); --删除表 drop table users; --创建表 ),xm ),sex ),birthday date,sal ,)); ),cnmae )); ...

  8. 将 varchar 值转换为 JDBC 数据类型 DATE 时发生错误。

    问题是: 我是这样解决的  : 网上的 转型方法 并不好使 ,我想了想 可能是由于返回值是String  我 就成功的解决错误了  ..下面是关于原理的讲解肯定方法不唯一   至于错误,的产生,这个肯 ...

  9. (原)python中使用plt.show()时显示图像

    转载请注明出处: http://www.cnblogs.com/darkknightzh/p/6039667.html 参考网址: http://matplotlib.org/users/shell. ...

  10. javascript数组方法鉴赏一

    创建数组 如果你习惯了用 new 来实例化对象的形式,那么在js中一定会疑惑,可选的参数数量代表的意义截然不同. new Array(size);//传一个参数的时候分两种情况,size是正整数时代表 ...