opencv3学习:reshape函数

摘要:
在opencv中,重塑函数更有趣。它不仅可以改变矩阵的通道数,还可以串行化矩阵元素。这是一个非常有用的功能。Intmain(){Matdata=Mat;//设置矩阵cout<“行数:”<<data.rrows<<endl;cout<”列数:“<<data.cols<<endr;cout>”channel:“<<data.channels()<<end l;cout<<endt;Matdst=data.shape(0,1);cout=“行数”<<dst.rrows><endl”cout<列数:system;return1;}第二个更改:保留通道数不变,序列化矩阵的N行和1列的列向量。我们知道,在matlab中,它是列序列化,即从上到下,从左到右。opencv怎么样?Intmain(){Matdata=;//矩阵cout的2行和3列˂˂data˂˂endl;Matdst1=data.reshape(0,6);//通道不变,序列向量cout˂˂endl˂˂dst1˂˂end1;Matdst2=data.reshape(0,1);//无论发生什么变化,通道不变,并且序列向量cout˂˂end l˂˂˂dst2˂˂end l;system;return1;}从结果中

在opencv中,reshape函数比较有意思,它既可以改变矩阵的通道数,又可以对矩阵元素进行序列化,非常有用的一个函数。

函数原型:

C++: Mat Mat::reshape(int cn, int rows=0) const

参数比较少,但设置的时候却要千万小心。

cn: 表示通道数(channels), 如果设为0,则表示保持通道数不变,否则则变为设置的通道数。

rows: 表示矩阵行数。 如果设为0,则表示保持原有的行数不变,否则则变为设置的行数。

首先设置一个初始矩阵:一个20行30列1通道的一个矩阵

int main()
{
    Mat data = Mat(20, 30, CV_32F);  //设置一个20行30列1通道的一个矩阵
    cout << "行数: " << data.rows << endl;
    cout << "列数: " << data.cols << endl;
    cout << "通道: " << data.channels() << endl;
    system("pause");
    return 1;
}

输出:

opencv3学习:reshape函数第1张

第一次变化:通道数不变,将矩阵序列化1行N列的行向量。

int main()
{
    Mat data = Mat(20, 30, CV_32F);  //设置一个20行30列1通道的一个矩阵
    cout << "行数: " << data.rows << endl;
    cout << "列数: " << data.cols << endl;
    cout << "通道: " << data.channels() << endl;
    cout << endl;
    Mat dst = data.reshape(0, 1);
    cout << "行数: " << dst.rows << endl;
    cout << "列数: " << dst.cols << endl;
    cout << "通道: " << dst.channels() << endl;
    system("pause");
    return 1;
}

opencv3学习:reshape函数第2张

第二次变化:通道数不变,将矩阵序列化N行1列的列向量。

int main()
{
    Mat data = Mat(20, 30, CV_32F);  //设置一个20行30列1通道的一个矩阵
    cout << "行数: " << data.rows << endl;
    cout << "列数: " << data.cols << endl;
    cout << "通道: " << data.channels() << endl;
    cout << endl;
    Mat dst = data.reshape(0, data.rows*data.cols);
    cout << "行数: " << dst.rows << endl;
    cout << "列数: " << dst.cols << endl;
    cout << "通道: " << dst.channels() << endl;
    system("pause");
    return 1;
}

opencv3学习:reshape函数第3张

 可见,序列成列向量比行向量要麻烦一些,还得去计算出需要多少行。但我们可以先序列成行向量,再转置

    Mat dst = data.reshape(0, 1);      //序列成行向量
    Mat dst = data.reshape(0, 1).t();  //序列成列向量

第三次变化:通道数由1变为2,行数不变。

int main()
{
    Mat data = Mat(20, 30, CV_32F);  //设置一个20行30列1通道的一个矩阵
    cout << "行数: " << data.rows << endl;
    cout << "列数: " << data.cols << endl;
    cout << "通道: " << data.channels() << endl;
    cout << endl;
    Mat dst = data.reshape(2, 0);
    cout << "行数: " << dst.rows << endl;
    cout << "列数: " << dst.cols << endl;
    cout << "通道: " << dst.channels() << endl;
    system("pause");
    return 1;
}

opencv3学习:reshape函数第4张

从结果可以看出,列数被分出一半,放在第二个通道里去了。

同理,如果通道数由1变为3,行数不变。则每通道的列数变为原来的三分之一。

需要注意的是,如果行保持不变,改变的通道数一定要能被列数整除,否则会出错。

第四次变化:通道数由1变为2,行数变为原来的五分之一。

int main()
{
    Mat data = Mat(20, 30, CV_32F);  //设置一个20行30列1通道的一个矩阵
    cout << "行数: " << data.rows << endl;
    cout << "列数: " << data.cols << endl;
    cout << "通道: " << data.channels() << endl;
    cout << endl;
    Mat dst = data.reshape(2, data.rows/5);
    cout << "行数: " << dst.rows << endl;
    cout << "列数: " << dst.cols << endl;
    cout << "通道: " << dst.channels() << endl;
    system("pause");
    return 1;
}

opencv3学习:reshape函数第5张

可见,不管怎么变,都遵循这样一个等式:

变化之前的  rows*cols*channels = 变化之后的 rows*cols*channels

我们只能改变通道数和行数,列数不能改变,它是自动变化的。

但是要注意的是,在变化的时候,要考虑到是否整除的情况。如果改变的数值出现不能整除,就会报错。

最后,我们再验证一下:opencv在序列化的时候是行序列化还是列序列化呢?

我们知道,在matlab里面,是列序列化, 即取值为从上到下,从左到右,opencv又是怎么样的呢

int main()
{
    Mat data = (Mat_<int>(2, 3) << 1, 2, 3, 10, 20, 30);  //2行3列的矩阵
    cout << data << endl;
    Mat dst1 = data.reshape(0, 6);   //通道不变,序列成列向量
    cout <<endl<< dst1 << endl;
    Mat dst2 = data.reshape(0, 1);   //通道不变,序列成行向量
    cout << endl << dst2 << endl;
    system("pause");
    return 1;
}

opencv3学习:reshape函数第6张

从结果看出,不管是变化成行向量还是列向量,opencv都是行序列化,即从左到右,从上到下,与matlab是不一样的。

简单的一个函数,功能却很强大!你会用了吗

免责声明:文章转载自《opencv3学习:reshape函数》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇[DM8168]Linux下控制GPIO控制12864液晶屏(ST7565控制器)并发调度的可串行性下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

基于java的OpenCV安装和配置

目录 OpenCV简介 OpenCV下载安装 eclipse里引用jar包和配置 OpenCV简介 OpenCV是一个基于BSD许可(开源)发行的跨平台计算机视觉库,可以运行在Linux、Windows、Android和Mac OS操作系统上。它轻量级而且高效——由一系列 C 函数和少量 C++ 类构成,同时提供了Python、Ruby、MATLA...

配置opencv cmake

第一种使用 find_package的方法示例代码如下:# 声明要求的 cmake 最低版本cmake_minimum_required( VERSION 2.8 ) # 声明一个 cmake 工程project( pro ) # 设置编译模式set( CMAKE_BUILD_TYPE "Debug" ) #添加OPENCV库#指定OpenCV版本,代码如...

项目实战:Qt+OpenCV图像处理与识别算法平台

若该文为原创文章,未经允许不得转载原博主博客地址:https://blog.csdn.net/qq21497936原博主博客导航:https://blog.csdn.net/qq21497936/article/details/102478062本文章博客地址:https://blog.csdn.net/qq21497936/article/details...

基于移动最小二乘的图像变形和曲线拟合

在最近的项目中经常遇到给出几个点需要拟合出一条曲线。 在离散的点云中,求曲线曲面拟合,不能简单地连接这些点,如果知道曲线曲面的形式,如为二次曲线等,可以简单地使用最小二乘法估计参数;但如果曲线曲面形式未知,可以使用移动最小二乘法或者主曲线方法。 转载:https://blog.csdn.net/liumangmao1314/article/details/...

OpenCV白平衡算法之灰度世界法(消除RGB受光照影响)

在用OpenCV对图像进行处理时,利用颜色定位是常常会接触到的方法,但RGB受光照影响比较严重,转换到HSV XYZ等空间也解决不了时, 可以用白平衡算法进行修正,使其发黄、发蓝、发红的照片更加趋于自然光下的图像。(转摘请说明来源) 程序代码示例如下: 1 //该代码实现白平衡算法中的灰度世界法,能有效改善图像发红发蓝发绿的现象; 2 3 #includ...

图像处理之基础---滤波器 高斯滤波

引用 keendawn 的 高斯(核)函数简介1函数的基本概念 所谓径向基函数 (Radial Basis Function 简称 RBF), 就是某种沿径向对称的标量函数。 通常定义为空间中任一点x到某一中心xc之间欧氏距离的单调函数 , 可记作 k(||x-xc||), 其作用往往是局部的 , 即当x远离xc时函数取值很小。最常用的径向基函数是高斯核...