Mat类的介绍与图像的读写

Mat类的介绍与图像的读写


1. Mat类的介绍

opencv2.x采用全行的图像数据结构Mat。Mat结构不需要我们为其手动开辟空间,也不需要立即释放存储空间,Mat类能够自动管理内存。Mat类由矩阵头和指向存储所有像素值的矩阵真的指针构成。Mat类表示一个n维的密集数值单通道或多通道数组,它可用于存储实数或负数值的向量和矩阵/灰度或彩色图像/体素/向量场/点云/张量/直方图等。

Mat类代码如下:

class CV_EXPORTS Mat
{
public:
	// 标志位
	int flags;
	// 数组的维数
	int dims;
	// 行数和列数
	int rows, cols;
	// 指向实际存储数据的指针
	uchar* data;
	// 指针的引用计数器,当阵列指向用户分配的数据时,指针为零
	int* refcount;
}

矩阵的头是一个固定大小的结构,而它所指向的数据存储于data指针指向的内存空间。

2. Mat类的构造函数与使用

Mat::Mat() 无参构造函数
Mat::Mat(int rows, cols) 创建行数为rows,列数为cols,类型为type的图像
Mat::Mat(Size size, int type) 创建大小为size,类型为type的图像
Mat::Mat(int rows, int cols, int type, const Scaler& s) 创建行数为rows,列数为cols,类型为type的图像,并将图像所有元素初始化为s
Mat::Mat(Size size, int type, const Scaler& s) 创建大小为size,类型为type的图像,并将所有元素初始化为s
Mat::Mat(const Mat& m) 将m复制给新创建的对象,此处并不会对象相数据进行复制,二者公用图像数据,即二者的data指针指向同一个地方

这里的type是用来指定图像的数据类型和同道数,常用的type由下表列出

type类型 含义
CV_8UC1 单通道图像,每个像素为8位无符号数
CV_16SC1 单通道图像,每个像素点为16位有符号数
CV_64FC3 3通道图像,每个像素点数据类型为64位浮点数

Mat使用举例

#include <iostream>
#include <opencv2/imgproc/imgproc.hpp>
#include <opencv2/highgui/highgui.hpp>

using namespace std;

int main()
{
    cv::Mat image1;
    cv::Mat image2(6, 6, CV_8UC1);
    cv::Mat image3(cv::Size(7, 7), CV_8UC3);
    cv::Mat image4(8, 8, CV_32FC2, cv::Scalar(1, 3));
    cv::Mat image5(cv::Size(9, 9), CV_8UC3, cv::Scalar(1,2,3));
    cv::Mat image6(image2);

    std::cout << "image1" << endl;
    std::cout << image1 << std::endl;
    std::cout << "image2" << endl;
    std::cout << image2 << std::endl;
    std::cout << "image3" << endl;
    std::cout << image3 << std::endl;
    std::cout << "image4" << endl;
    std::cout << image4 << std::endl;
    std::cout << "image5" << endl;
    std::cout << image5 << std::endl;
    std::cout << "image6" << endl;
    std::cout << image6 << std::endl;

    return 0;
}

3. Mat的一些常用成员函数

函数 功能
Mat::row 创建一个具有指定了矩阵头中函数的参数的矩阵
Mat::col 创建一个具有指定了矩阵头中列数的参数的矩阵
Mat::rowRange 为指定的行span创建了一个新的矩阵头,可取指定行区间元素
Mat::colRange 为指定的列span创建一个矩阵头,可取指定的列区间元素
Mat::clone 创建一个数组及其基础数据的完整副本
Mat::copyTo 把矩阵复制到另一个矩阵中
Mat::convertTo 在缩放或不缩放的情况下转换成另一种数据类型
Mat::channels 返回矩阵的通道数
Mat::empty 如果数组中没有元素,则返回真
Mat::at 返回对指定数据元素的引用

举例:

#include <iostream>
#include <opencv2/imgproc/imgproc.hpp>
#include <opencv2/highgui/highgui.hpp>

using namespace std;
using namespace cv;

int main()
{
    Mat Image1( 10, 8, CV_8UC1, Scalar(5) );
    cout << "Image1 row: " << Image1.rows << endl;
    cout << "Image1 col: " << Image1.cols << endl;

    cout << Image1.rowRange(1, 3) << endl;
    cout << Image1.colRange(2, 4) << endl;

    Mat Image2( 8, 8, CV_32FC2, Scalar(1, 5) );
    Image2.create( 10, 10, CV_8UC3 );
    cout << "Image2 channels: " << Image2.channels() << endl;
    Image2.convertTo( Image2, CV_32F );
    cout << "Image2 depth: " << Image2.depth() << endl;

    Mat Image3 = Mat::zeros( Image2.rows, Image2.cols, CV_8UC1 );
    Image1.row(4) = Image1.row(5) * 2;
    cout << Image1 << endl;

    Mat Image4 = Image1.col(4);
    cout << Image4 << endl;

    Image1.col(1).copyTo(Image4);
    cout << Image4 << endl;

    return 0;
}

4. 图像的读写与显示

opencv提供了类似MATLAB中相类似的imread/imwrite/imshow等函数,具体情况如下

Mat imread( const string& filename, int flags=1 )

imread用于读取图像
filename: 图像名
flags:表示读取图像的颜色类型,默认参数为1,函数返回3通道图像。参数flags有如下几种常用设置方式:

flags 含义
CV_LOAD_IMAGE_ANYDEPTY 返回图像真正的位宽(16bit或者32bit),否则返回8bit
CV_LOAD_IMAGE_COLOR 返回单色图像
CV_LOAD_IMAGE_GRAYSCALE 返回通道图像
<0 不对图像进行通道转换
0 返回单通道图像
>0 强制返回3通道图像
bool imwrite( const string& filename, InputArray img, const vector<int>& params=vector<int>() )

filename:写入文件的格式及文件扩展名
img:表示待写入的图像数据
params:表示文件格式的一些信息

读写图像举例

#include <iostream>
#include <opencv2/core/core.hpp>
#include <opencv2/highgui/highgui.hpp>
#include <opencv2/imgproc/imgproc.hpp>

using namespace std;
using namespace cv;

int main()
{
    // 读取图像数据
    Mat src_image = imread("./android.jpg");

    // 判断是否读取成功
    if ( src_image.empty() )
    {
        return -1;
    }

    Mat src_gray;

    // rgb2gray
    cvtColor(src_image, src_gray, COLOR_RGB2GRAY);

    imwrite("src_gray.png", src_gray);
    imshow("src_gray", src_gray);

    Mat blur_dst_image;

    // 图像均值滤波
    blur( src_gray, blur_dst_image, Size(5, 5), Point(-1, -1) );
    imshow("blur_dst_image", blur_dst_image);

    // 均值滤波后的图像写如到本地硬盘
    imwrite("blur_dst_image.png", blur_dst_image);
    waitKey(0);

    return 0;
}

Mat类的介绍与图像的读写