Opencv中Mat矩阵相乘——点乘、dot、mul运算详解

Stella981
• 阅读 1998

Opencv中Mat矩阵相乘——点乘、dot、mul运算详解

2016年09月02日 00:00:36 -牧野- 阅读数:59593 标签: Opencv矩阵相乘点乘dotmul 更多

个人分类: OpenCV

所属专栏: OpenCV从入门到转行

版权声明:本文为博主原创文章,转载请注明出处。 https://blog.csdn.net/dcrmg/article/details/52404580

Mat矩阵点乘——A*B

Opencv重载了运算符“*”,姑且称之为Mat矩阵“点乘”,其中一个重载声明为:

CV_EXPORTS MatExpr operator * (const Mat& a, const Mat& b);

点乘说明:

1.  A*B是以数学运算中矩阵相乘的方式实现的,即Mat矩阵A和B被当做纯粹的矩阵做乘法运算,这就要求A的列数等       于B的行数时,才能定义两个矩阵相乘。如A是m×n矩阵,B是n×p矩阵,它们的乘积AB是一个m×p矩阵。

Opencv中Mat矩阵相乘——点乘、dot、mul运算详解

如上图所示,C=AB。C中第i行第j列所在元素C(i,j)等于A中第i行所有元素跟B中第j列所有元素一一对应的乘积之和。

更具有代表性的的:对于A、B都是2行2列矩阵的情况:

Opencv中Mat矩阵相乘——点乘、dot、mul运算详解

Opencv验证:

定义两个Mat矩阵A和B点乘,A为2行3列,B为3行2列:

  1. #include "core/core.hpp"

  2. #include "iostream"

  3. using namespace std;

  4. using namespace cv;

  5. int main(int argc,char *argv[])

  6. {

  7. Mat A=Mat::ones( 2,3,CV_32FC1);

  8. Mat B=Mat::ones( 3,2,CV_32FC1);

  9. Mat AB;

  10. A.at< float>(0,0)=1;

  11. A.at< float>(0,1)=2;

  12. A.at< float>(0,2)=3;

  13. A.at< float>(1,0)=4;

  14. A.at< float>(1,1)=5;

  15. A.at< float>(1,2)=6;

  16. B.at< float>(0,0)=1;

  17. B.at< float>(0,1)=2;

  18. B.at< float>(1,0)=3;

  19. B.at< float>(1,1)=4;

  20. B.at< float>(2,0)=5;

  21. B.at< float>(2,1)=6;

  22. AB=A*B;

  23. cout<<"A=\n"<<A<<endl<<endl;

  24. cout<<"B=\n"<<B<<endl<<endl;

  25. cout<<"AB=\n"<<AB<<endl<<endl;

  26. system( "pause");

  27. }

输出:

Opencv中Mat矩阵相乘——点乘、dot、mul运算详解

务必保证两个Mat矩阵中第一个矩阵A的列数等于第二个矩阵B的行数。

2.  参与点乘的两个Mat矩阵的数据类型(type)只能是 CV_32F、 CV_64FC1、 CV_32FC2、 CV_64FC2 这4种类        型中的一种。若选用其他类型,比如CV_8UC1,编译器会报错:

Opencv中Mat矩阵相乘——点乘、dot、mul运算详解

Mat矩阵dot——A.dot(B)

Opencv中.dot操作才算得上是真正的“点乘”,A.dot(B)操作相当于数学向量运算中的点乘,也叫向量的内积、数量积。

函数声明:

  1. //! computes dot-product

  2. double dot(InputArray m) const;

dot说明:

1.  对两个向量执行点乘运算,就是对这两个向量对应位一一相乘之后求和的操作,点乘的结果是一个标量。 

对于向量a和向量b:

                                     Opencv中Mat矩阵相乘——点乘、dot、mul运算详解                       Opencv中Mat矩阵相乘——点乘、dot、mul运算详解

a和b的点积公式为:

Opencv中Mat矩阵相乘——点乘、dot、mul运算详解

要求向量a和向量b的行列数相同。

Mat矩阵的dot方法扩展了一维向量的点乘操作,把整个Mat矩阵扩展成一个行(列)向量,之后执行向量的点乘运算,仍然要求参与dot运算的两个Mat矩阵的行列数完全一致。

2.  dot方法声明中显示返回值是double,所以A.dot(B)结果是一个double类型数据,不是Mat矩阵,不能把A.dot(B)结       果赋值给Mat矩阵!

Opencv验证:

  1. #include "core/core.hpp"

  2. #include "iostream"

  3. using namespace std;

  4. using namespace cv;

  5. int main(int argc,char *argv[])

  6. {

  7. Mat A=Mat::ones( 2,3,CV_8UC1);

  8. Mat B=Mat::ones( 2,3,CV_8UC1);

  9. A.at( 0,0)=1;

  10. A.at( 0,1)=2;

  11. A.at( 0,2)=3;

  12. A.at( 1,0)=4;

  13. A.at( 1,1)=5;

  14. A.at( 1,2)=6;

  15. B.at( 0,0)=1;

  16. B.at( 0,1)=2;

  17. B.at( 0,2)=3;

  18. B.at( 1,0)=4;

  19. B.at( 1,1)=5;

  20. B.at( 1,2)=6;

  21. double AB=A.dot(B);

  22. cout<<"A=\n"<<A<<endl<<endl;

  23. cout<<"B=\n"<<B<<endl<<endl;

  24. cout<<"double类型的AB=\n"<<AB<<endl<<endl;

  25. system( "pause");

  26. }

运行结果:

Opencv中Mat矩阵相乘——点乘、dot、mul运算详解

若对AB声明为Mat,则在编译阶段就会报错。

3.  dot操作不对参与运算的矩阵A、B的数据类型做要求,CV_8UC1、CV_32FC1等,可以是任何Opencv定义的类         型,如在2中使用的就是CV_8UC1。

4.  若参与dot运算的两个Mat矩阵是多通道的,则计算结果是所有通道单独计算各自.dot之后,再累计的和,结果仍是一个double类型数据。

Mat矩阵mul——A.mul(B)

Opencv中mul会计算两个Mat矩阵对应位的乘积,所以要求参与运算的矩阵A的行列和B的行列数一致。计算结果是跟A或B行列数一致的一个Mat矩阵。

Opencv中mul声明:

  1. //! per-element matrix multiplication by means of matrix expressions

  2. MatExpr mul(InputArray m, double scale=1) const;

以简单的情况为例,对于2*2大小的Mat矩阵A和B:

Opencv中Mat矩阵相乘——点乘、dot、mul运算详解

对A和B执行mul运算:

Opencv中Mat矩阵相乘——点乘、dot、mul运算详解

mul说明:

1.  mul操作不对参与运算的两个矩阵A、B有数据类型上的要求,但要求A,B类型一致,不然报错;

2.  Mat AB=A.mul(B),若声明AB时没有定义AB的数据类型,则默认AB的数据类型跟A和B保存一致;

3.  若AB精度不够,可能产生溢出,溢出的值被置为当前精度下的最大值;

Opencv验证:

  1. #include "core/core.hpp"

  2. #include "iostream"

  3. using namespace std;

  4. using namespace cv;

  5. int main(int argc,char *argv[])

  6. {

  7. Mat A=Mat::ones( 2,3,CV_8UC1);

  8. Mat B=Mat::ones( 2,3,CV_8UC1);

  9. A.at( 0,0)=60;

  10. A.at( 0,1)=2;

  11. A.at( 0,2)=3;

  12. A.at( 1,0)=4;

  13. A.at( 1,1)=5;

  14. A.at( 1,2)=6;

  15. B.at( 0,0)=60;

  16. B.at( 0,1)=2;

  17. B.at( 0,2)=3;

  18. B.at( 1,0)=4;

  19. B.at( 1,1)=5;

  20. B.at( 1,2)=6;

  21. Mat AB=A.mul(B);

  22. cout<<"A=\n"<<A<<endl<<endl;

  23. cout<<"B=\n"<<B<<endl<<endl;

  24. cout<<"AB=\n"<<AB<<endl<<endl;

  25. system( "pause");

  26. }

输出:

Opencv中Mat矩阵相乘——点乘、dot、mul运算详解

AB中第一个元素应该为60*60=360,但AB默认的类型为CV_8UC1,即最大值只能是255;所以执行mul运算一定要定义AB足够的精度,防止溢出。

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Easter79 Easter79
3年前
swap空间的增减方法
(1)增大swap空间去激活swap交换区:swapoff v /dev/vg00/lvswap扩展交换lv:lvextend L 10G /dev/vg00/lvswap重新生成swap交换区:mkswap /dev/vg00/lvswap激活新生成的交换区:swapon v /dev/vg00/lvswap
待兔 待兔
6个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
Jacquelyn38 Jacquelyn38
3年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
Wesley13 Wesley13
3年前
Java获得今日零时零分零秒的时间(Date型)
publicDatezeroTime()throwsParseException{    DatetimenewDate();    SimpleDateFormatsimpnewSimpleDateFormat("yyyyMMdd00:00:00");    SimpleDateFormatsimp2newS
Wesley13 Wesley13
3年前
mysql设置时区
mysql设置时区mysql\_query("SETtime\_zone'8:00'")ordie('时区设置失败,请联系管理员!');中国在东8区所以加8方法二:selectcount(user\_id)asdevice,CONVERT\_TZ(FROM\_UNIXTIME(reg\_time),'08:00','0
Wesley13 Wesley13
3年前
00:Java简单了解
浅谈Java之概述Java是SUN(StanfordUniversityNetwork),斯坦福大学网络公司)1995年推出的一门高级编程语言。Java是一种面向Internet的编程语言。随着Java技术在web方面的不断成熟,已经成为Web应用程序的首选开发语言。Java是简单易学,完全面向对象,安全可靠,与平台无关的编程语言。
Wesley13 Wesley13
3年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
Python进阶者 Python进阶者
1年前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这
Oracle 分组与拼接字符串同时使用
SELECTT.,ROWNUMIDFROM(SELECTT.EMPLID,T.NAME,T.BU,T.REALDEPART,T.FORMATDATE,SUM(T.S0)S0,MAX(UPDATETIME)CREATETIME,LISTAGG(TOCHAR(