前言


  因为对图像方面感兴趣,所以有空学学OpenCV的使用,并且希望以此为引子,带领自己入门图像领域。

  先post上几个参考网站,上面有完整源码:

  因为这么多资源,所以就不贴完整代码,这重点讲解某部分,方便自己以后回来查询。

Mat - 基本的图像容器


Mat

  在以前,opencv使用的是C结构,IplImage。但是使用这个结构有一个缺点就是你需要注意内存的申请和销毁。幸运的是,在C++我们可以使用一种更智能的结构,Mat。Mat会自动申请内存和销毁。

  Mat由基本的两部分组成:矩阵头(包含图片信息,例如矩阵大小,存储方法等)和一个指向包含像素点信息的指针。矩阵头部大小是常数,但是矩阵的大小却各不相同。

 Mat A, C;  // 只建立头部
A = imread(fname, CV_LOAD_IMAGE_COLOR); Mat B(A); // 调用copy构造函数
C = A; // 调用assign函数

  上面的所有对象都指向同一个矩阵,只是头部不一样而已。如果使用其中一个对象改变图像内容,所有指向这个矩阵的对象都会受影响。copy和assign只是复制头部的一些信息。

  我们可以调用其它方法实现深复制:

 Mat F = A.clone();
Mat G;
A.copyTo(G);

显式创建Mat

  我们可以使用 imwrite() 函数来把一个矩阵写入到图片文件。但是为了调试方便,我们还可以使用<<输出(仅适用于二维矩阵)。

  下面是创建Mat对象的各种方法:

  • Mat()构造器

     Mat M(, , CV_8UC3, Scalar(, , )); // CV_[多少位][有符号or无符号][类型前缀]C[通道数]
    cout << M <<endl;
    // [0, 0, 255, 0, 0, 255;
    // 0, 0, 255, 0, 0, 255
  • 使用C\C++数组构造
     int sz[] = {, 2,2};
    Mat L(, sz, CV_8UC(), Scalar::all()); // 3维的[2, 2, 2]的图像
  • 为已存在IplImage指针构建头部
     IplImage* img = cvLoadImage(fname);
    Mat mtx(img);
  • Create() 函数
     M.create(, , CV_8UC()); // 这种方法不能赋初值,只在中心分配内存时使用
    cout<< M << endl
  • Matlab风格的初始化
     Mat E = Mat::eye(, , CV_64F);
    Mat O = Mat::ones(, , CV_32F);
    Mat Z = Mat::zeros(, , CV_8UC1);
  • 逗号分隔的初始化小矩阵
     Mat C = (Mat_<double>(,) << , -, , -, , -, , -, );
  • 使用clone或copyTo。
     Mat RowClone = C.row(1).clone(); // randu(RowClone, Scalar::all(0), Scalar::all(255))可以在low和high之间随机

   

怎样遍历图片


  首先,我们可以使用一段代码计算程序执行的时间:

 double t = (double)getTickCount();
// do something
t = ((double)getTickCount() - t) / getTickFrequency();

  

图像的存储

  在RGB系统中,图像是这样存储的:(注意是BGR的形式,可以使用 isContinunous() 函数查看是否连续存放)

(下面以颜色空间缩减为例子说明)

C风格的读法

先用指针p指向一行,然后再p[j]形式读取

 int channels = I.channels();
int nRows = I.rows;
int nCols = I.cols * channels; if (I.isContinuous()) {
nCols = nCols * nRows;
nRows = ;
} int i, j;
uchar* p;
for (i=; i<nRows; ++i) {
for (j=; j<nCols; ++j) {
p = I.ptr<uchar>(i);
p[j] = table[p[j]]; //查表替换
}
}

迭代(安全)方法

迭代器从begin到end,使用(*it)[0]形式读取

const int channels = I.channels();
switch(channels)
{
case : {
MatIterator_<uchar> it, end;
for (it = I.begin<uchar>(), end=I.end<uchar>(); it != end; ++it)
*it = table[*it];
break;
}
case : {
MatIterator_<Vec3b> it, end;
for (it = I.begin<Vec3b>(), end=I.end<Vec3b>(); it != end; ++it)
(*it)[] = table[(*it)[]];
(*it)[] = table[(*it)[]];
(*it)[] = table[(*it)[]];
}
}

通过相关返回值的On-the-fly地址计算

先把矩阵转换成Mat_,再用_I(i, j)[0]形式读取

const int channels = I.channels();
switch(channels)
{
case :
{
for (int i=; i<I.rows; ++i)
for (int j=; j<I.cols; ++j)
I.at<uchar>(i, j) = table[I.at<uchar>(i, j)];
break;
}
case :
{
Mat_<Vec3b> _I = I;
for (int i=; i<I.rows; ++i)
for (int j=; j<I.cols; ++j) {
_I(i, j)[] = table[_I(i, j)[]];
_I(i, j)[] = table[_I(i, j)[]];
_I(i, j)[] = table[_I(i, j)[]];
}
I = _I;
break;
}
}

快速实现表替换

Mat lookUpTable(, , CV_8U);
uchar* p = lookUpTable.data;
for (int i=; i < ; ++i)
p[i] = table[i];
LUT(I, lookUpTable, I);

OpenCv 2.4.9 (一) Mat基础结构&如何遍历图片的更多相关文章

  1. OpenCV中图像的格式Mat 图像深度

    opencv中图像的格式Mat 有图像的定义,图像深度.类型格式等,其中Mat的参数depth为深度,深度反应出图像颜色像素值: 关于数据的储存:(转) Mat_<uchar>对应的是CV ...

  2. opencv图像处理时使用文件输入流ifstream批量读取图片

    简介: 在利用opencv进行图像处理时,通常需要批量读取图片,然后做相应的处理,我们可以用C++文件的输入流来进行图片的读取,这要求我们应该事先,将图片图片名生成txt文件,具体请参见之前的博文[u ...

  3. opencv基础知识------IplImage, CvMat, Mat 的关系和相互转换

    Mat,cvMat和IplImage这三种类型都可以代表和显示图像,但是,Mat类型侧重于计算,数学性较高,openCV对Mat类型的计算也进行了优化.而CvMat和IplImage类型更侧重于“图像 ...

  4. OpenCV几种访问cv::Mat数据的方法

    一般来说,如果是遍历数据的话用指针ptr比用at要快.特别是在debug版本下.因为debug中,OpenCV会对at中的坐标检查是否有溢出,这是非常耗时的. 代码如下 #include <op ...

  5. opencv 数据类型转换:CvArr, Mat, CvMat, IplImage, BYTE 转

    留着以后查询: http://blog.csdn.net/augusdi/article/details/8863820 一.Mat类型:矩阵类型,Matrix. 在openCV中,Mat是一个多维的 ...

  6. OpenCV学习(6) 文件和Mat之间的数据交换

          有时候为了便于调试算法,我们需要从文本文件或二进制文件中读取数据,并把数据放到相应的矩阵中去.我们通常可以通过下面的函数实现.   1.从二进制文件中读取数据.      新建一个txt文 ...

  7. opencv中的高维矩阵Mat

    本示例程序主要是通过实例演示高维Mat的寻址方式. //3,4分别表示行数.列数,所以3*4是一个页面的元素数,2表示有2个3*4 ,b=,c=; int size[]={a,b,c}; float* ...

  8. OpenCV成长之路(2):图像的遍历

    我们在实际应用中对图像进行的操作,往往并不是将图像作为一个整体进行操作,而是对图像中的所有点或特殊点进行运算,所以遍历图像就显得很重要,如何高效的遍历图像是一个很值得探讨的问题. 一.遍历图像的4种方 ...

  9. Opencv与Qt (一)之运行测试读取图片

    刚刚在vs上装好了QT和Opencv,试一下效果把. 我简单的创建了一个label,然后使用Opencv导入图像,因为Opencv导入图像是MAT格式的,在使用Qt的时候我们要把导入的图像转换成Qim ...

随机推荐

  1. 多层神经网络BP算法 原理及推导

    首先什么是人工神经网络?简单来说就是将单个感知器作为一个神经网络节点,然后用此类节点组成一个层次网络结构,我们称此网络即为人工神经网络(本人自己的理解).当网络的层次大于等于3层(输入层+隐藏层(大于 ...

  2. Oracle的一些命令

    sqlldr: 一般用于导入以任何后缀结束的文件,我这次就是因为要导入一张以.20160101为后缀的文件,当初简直束手无策 结合input.ctl使用,可以在DOS下使用,可以对一张表导入数十万,百 ...

  3. js正则表达式验证

    有时候会要验证自己写的正则表达式是否正确 所以写了这个小东西: demo:js正则表达式验证 html: <h3>绿色表示匹配,红色表示不匹配</h3> <label&g ...

  4. php正则表达式简介

    正则表达式简介 正则表达式Regular Expression,使用单个字符串来描述.匹配一系列符合某个句法规则的字符串.代表性书籍<正则表达式之道>里面有详细论述,本书籍为个人查阅之用力 ...

  5. 判断移动端设备: navigator.userAgent.toLowerCase()

    判断你的浏览设备: navigator.userAgent.toLowerCase(); (返回当前用户所使用的是什么浏览器,将获得的信息变成小写) function browserRedirect( ...

  6. 获取CPU系列号,硬盘系

    unit Secrity; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, For ...

  7. Eclipse中应用的调试

    作为编程人员,程序的调试是一项基本功.在不使用IDE的时候,程序的调试多数是通过日志或者输入语句(System.out.println)的方式.可以把程序运行的轨迹或者程序运行过程中的状态显示给用户, ...

  8. 用NetStream的appendBytes播放FLV

    public class MiniStream extends Sprite { private var _buffer:ByteArray = new ByteArray(); private va ...

  9. Codeforces Round #392 (Div. 2)-758D. Ability To Convert(贪心,细节题)

    D. Ability To Convert time limit per test 1 second Cmemory limit per test 256 megabytes input standa ...

  10. ADO.NET高级应用

    ADO.NET事务处理(4个步骤) 1.调用SqlConnection对象的BeginTransaction()方法,创建一个SqlTransaction对象,标志事务开始. 2.将创建的SqlTra ...