OpenCV中Mat的传值、传引用、clone()、copyto()

来源:互联网 发布:天猫淘宝 编辑:程序博客网 时间:2024/05/23 05:09

1、前提

在C++中,函数的参数,传值、传引用、传指针是有区别的,具体如下:

#include<iostream>#include <iomanip>using namespace std;void test1(int a){    a = a + 1;}void test2(int &a){    a = a + 1;}void test3(int *p){    *p = *p + 1;}#if 1void main(){    int a = 23;    int *p = &a;    int &b = a;    cout << endl << setw(4) << "a" << setw(4) << "*p" << setw(10) << "p" << setw(4) << "b" << endl;    cout << endl << "输出初始值:" << endl;    cout << setw(4) << a << setw(4) << *p << setw(10) << p << setw(4) << b << endl << endl;    test1(a);    cout << "传值:" << endl;    cout << setw(4) << a << setw(4) << *p << setw(10) << p << setw(4) << b << endl << endl;    test2(a);    cout << "传引用:" << endl;    cout << setw(4) << a << setw(4) << *p << setw(10) << p << setw(4) << b << endl << endl;    test3(p);    cout << "传指针:" << endl;    cout << setw(4) << a << setw(4) << *p << setw(10) << p << setw(4) << b << endl << endl;    system("pause");}#else#endif

结果如下:

这里写图片描述

这说明,传值不会改变实参,传引用和传指针都会。这是因为传值传的一个复制,不会改变原来的参数,而传引用和传指针都是在指向实参的一块内存上进行改动,改变的就是实参本身,所以会发生改变。

但是这里要说的是OpenCV中Mat数据结构在这上面的操作。

2、Mat

本文操作的原图为:
.

这里写图片描述

.

2.1

首先,colorReduce()函数是对图像颜色进行缩减,以下采用了传值和传引用的方式。

#include <opencv2/core/core.hpp>#include <opencv2/highgui/highgui.hpp>using namespace std;using namespace cv;void colorReduce_0(Mat image, int div){    int nl = image.rows;    int nc = image.cols * image.channels();    for (int j = 0; j < nl; j++)    {        uchar *data = image.ptr<uchar>(j);        for (int i = 0; i < nc; i++)        {            data[i] = data[i] / div * div + div / 2;            //data[i] = data[i] - data[i] % div + div / 2;        }    }}void colorReduce_1(Mat &image, int div){    int nl = image.rows;    int nc = image.cols * image.channels();    for (int j = 0; j < nl; j++)    {        uchar *data = image.ptr<uchar>(j);        for (int i = 0; i < nc; i++)        {            data[i] = data[i] / div * div + div / 2;            //data[i] = data[i] - data[i] % div + div / 2;        }    }}void main(){    Mat img0 = imread("groot.jpg");    Mat img00 = imread("groot.jpg");    colorReduce_0(img0, 64);    imshow("reduce0", img0);    colorReduce_1(img00, 64);    imshow("reduce1", img00);    waitKey(0);}

这里,两次分别是传值和传引用,结果实参都发生了改变,也就是说原图发生了改变。

这里写图片描述

这是因为传值时,只是将Mat这个结构体信息拷贝了一份,并没有拷贝指向图像的内存信息,操作仍然在同一内存中进行的,所以原图,也就是实参发生了改变。

2.2

然后现在说,如果不改变原图,来进行操作。

  • 一种是使用clone和copyto函数,这是图像的深拷贝,相当于重新创建了一份一模一样的图像。
  • 另一种是创建跟原图像大小、格式相同的空图,即空Mat,然后用原图的像素来填充新图。
#include <opencv2/core/core.hpp>#include <opencv2/highgui/highgui.hpp>using namespace std;using namespace cv;Mat colorReduce_2(Mat img,int div){    Mat image;    image = img.clone();    //img.copyTo(image);    int nl = image.rows;    int nc = image.cols * image.channels();    for (int j = 0; j < nl;j++)    {        uchar *data = image.ptr<uchar>(j);        for (int i = 0; i < nc;i++)        {            //data[i] = data[i] / div * div + div / 2;            data[i] = data[i] - data[i] % div + div / 2;        }    }    return image;}void colorReduce_3(const Mat &image, Mat result,int div){    int nl = image.rows;    int nc = image.cols * image.channels();    for (int j = 0; j < nl; j++)    {        const uchar *data_in = image.ptr<uchar>(j);        uchar *data_out = result.ptr<uchar>(j);        for (int i = 0; i < nc; i++)        {            data_out[i] = data_in[i] / div * div + div / 2;        }    }}void main(){    Mat img = imread("groot.jpg");    Mat img2 = colorReduce_2(img, 64);    imshow("reduce2", img2);    imshow("groot1", img);    Mat result;    result.create(img.rows, img.cols, img.type());    colorReduce_3(img, result, 64);    imshow("reduce3", result);    imshow("groot2", img);    waitKey(0);}

可以看出,这里的两次,原图都没有发生变化。

这里写图片描述

这里写图片描述


原创粉丝点击