您的位置:首页 > 运维架构

OpenCv:Mat矩阵的初始化

2015-10-11 22:06 435 查看
在研究Mat矩阵的初始化的时候,发现其不能像Matx轻量级矩阵那样,直接利用数组来进行初始化,而是利用一个内部的变量类型:Scalar来进行初始化的。参考手册中的构造函数如下所示:
(1) Mat::Mat()(2) Mat::Mat(int rows, int cols, int type)(3) Mat::Mat(Size size, int type)(4) Mat::Mat(int rows, int cols, int type, constScalar& s)(5) Mat::Mat(Size size, int type, constScalar& s)(6) Mat::Mat(const Mat& m)(7) Mat::Mat(int rows, int cols, int type, void* data, size_t step=AUTO_STEP)(8) Mat::Mat(Size size, int type, void* data, size_t step=AUTO_STEP)(9) Mat::Mat(const Mat& m, const Range& rowRange, const Range& colRange)(10) Mat::Mat(const Mat& m, const Rect& roi)(11) Mat::Mat(const CvMat* m, bool copyData=false)(12) Mat::Mat(const IplImage* img, bool copyData=false)(13) template<typename T, int n> explicit Mat::Mat(const Vec<T, n>& vec, bool copyData=true)(14) template<typename T, int m, int n> explicit Mat::Mat(const Matx<T, m, n>& vec, bool copyData=true)(15) template<typename T> explicit Mat::Mat(const vector<T>& vec, bool copyData=false)(16) Mat::Mat(const MatExpr& expr)(17) Mat::Mat(int ndims, const int* sizes, int type)(18) Mat::Mat(int ndims, const int* sizes, int type, constScalar& s)(19) Mat::Mat(int ndims, const int* sizes, int type, void* data, const size_t* steps=0)(20) Mat::Mat(const Mat& m, const Range* ranges)
似乎这样的矩阵让我们用起来感觉很不顺手,不过接着往下读参考手册,你会发现这样一个例子:

Mat H(100, 100, CV_64F);

       for(int i = 0; i < H.rows; i++)

            for(int j = 0; j < H.cols; j++)

                  H.at<double>(i,j)=1./(i+j+1);

通过这个例子,我们尝试来给Mat类型初始化。

第一种改进:(直接由以上例子仿制而来)

//用一个常数来初始化矩阵

void InitMat(Mat& m,float t)

{

 for(int i=0;i<m.rows;i++)

  for(int j=0;j<m.cols;j++)

   m.at<float>(i,j)=t;

}

主程序:

int _tmain(int argc, _TCHAR* argv[])

{

 //Mat矩阵中的初始化,可以利用Mat::at()来完成

 Mat M0(3,3,CV_32F);

 InitMat(M0,3);

}

第二种改进:

//用一个一维数组来初始化矩阵

void InitMat(Mat& m,float* num)

{

 for(int i=0;i<m.rows;i++)

  for(int j=0;j<m.cols;j++)

   m.at<float>(i,j)=*(num+i*m.rows+j);

}

主程序:

int _tmain(int argc, _TCHAR* argv[])

{

 //Mat矩阵中的初始化,可以利用Mat::at()来完成

float m0[]={ 1,2,3,

     6,5,4,

     7,8,9 };

 Mat M0(3,3,CV_32F);

 InitMat(M0,m0);

}

 

第三种改进:

//希望利用二维数组来初始化

void InitMat(Mat& m,float(*p)[3])

{

 for(int i=0;i<m.rows;i++)

 for(int j=0;j<m.cols;j++)

   m.at<float>(i,j)=*(*(p+i)+j);

}

主程序:

int _tmain(int argc, _TCHAR* argv[])

{

 //Mat矩阵中的初始化,可以利用Mat::at()来完成

float m[][3]={ 1,2,3,

     6,5,4,

     7,8,9 };

 Mat M0(3,3,CV_32F);

 InitMat(M0,m);

}

 

然而,注意在传递二维数组的时候,第三种方法的局限性在于必须要知道每一维度的元素个数,其实可以将二维数组,例如a[2][3]的数组名称a直接转换成一个float指针传递进去,当成一位数组使用。即,

主程序:

int _tmain(int argc, _TCHAR* argv[])

{

 //Mat矩阵中的初始化,可以利用Mat::at()来完成

float m[][3]={ 1,2,3,

     6,5,4,

     7,8,9 };

 Mat M0(3,3,CV_32F);

 InitMat(M0,(float*)m);

}

 注:

数组和指针在参数传递时的对应关系:

                                      实参                     形参

数组的数组:               char a[8][10]     char(*p)[10]

指针的数组:               char *a[10]        char **p

数组指针(行指针): char(*a)[8]         char(a)[8]

指针的指针:               char **a            char**a

 

结果运行截图如下:

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: