Opencv中Mat矩阵相乘——点乘、dot、mul运算详解
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矩阵。
如上图所示,C=AB。C中第i行第j列所在元素C(i,j)等于A中第i行所有元素跟B中第j列所有元素一一对应的乘积之和。
更具有代表性的的:对于A、B都是2行2列矩阵的情况:
Opencv验证:
定义两个Mat矩阵A和B点乘,A为2行3列,B为3行2列:
- #include "core/core.hpp"
- #include "iostream"
- using namespace std;
- using namespace cv;
- int main(int argc,char *argv[])
- {
- Mat A=Mat::ones(2,3,CV_32FC1);
- Mat B=Mat::ones(3,2,CV_32FC1);
- Mat AB;
- A.at<float>(0,0)=1;
- A.at<float>(0,1)=2;
- A.at<float>(0,2)=3;
- A.at<float>(1,0)=4;
- A.at<float>(1,1)=5;
- A.at<float>(1,2)=6;
- B.at<float>(0,0)=1;
- B.at<float>(0,1)=2;
- B.at<float>(1,0)=3;
- B.at<float>(1,1)=4;
- B.at<float>(2,0)=5;
- B.at<float>(2,1)=6;
- AB=A*B;
- cout<<"A=\n"<<A<<endl<<endl;
- cout<<"B=\n"<<B<<endl<<endl;
- cout<<"AB=\n"<<AB<<endl<<endl;
- system("pause");
- }
输出:
务必保证两个Mat矩阵中第一个矩阵A的列数等于第二个矩阵B的行数。
2. 参与点乘的两个Mat矩阵的数据类型(type)只能是 CV_32F、 CV_64FC1、 CV_32FC2、 CV_64FC2 这4种类 型中的一种。若选用其他类型,比如CV_8UC1,编译器会报错:
Mat矩阵dot——A.dot(B)
Opencv中.dot操作才算得上是真正的“点乘”,A.dot(B)操作相当于数学向量运算中的点乘,也叫向量的内积、数量积。
函数声明:
- //! computes dot-product
- double dot(InputArray m) const;
dot说明:
1. 对两个向量执行点乘运算,就是对这两个向量对应位一一相乘之后求和的操作,点乘的结果是一个标量。
对于向量a和向量b:
a和b的点积公式为:
要求向量a和向量b的行列数相同。
Mat矩阵的dot方法扩展了一维向量的点乘操作,把整个Mat矩阵扩展成一个行(列)向量,之后执行向量的点乘运算,仍然要求参与dot运算的两个Mat矩阵的行列数完全一致。
2. dot方法声明中显示返回值是double,所以A.dot(B)结果是一个double类型数据,不是Mat矩阵,不能把A.dot(B)结 果赋值给Mat矩阵!
Opencv验证:
- #include "core/core.hpp"
- #include "iostream"
- using namespace std;
- using namespace cv;
- int main(int argc,char *argv[])
- {
- Mat A=Mat::ones(2,3,CV_8UC1);
- Mat B=Mat::ones(2,3,CV_8UC1);
- A.at<uchar>(0,0)=1;
- A.at<uchar>(0,1)=2;
- A.at<uchar>(0,2)=3;
- A.at<uchar>(1,0)=4;
- A.at<uchar>(1,1)=5;
- A.at<uchar>(1,2)=6;
- B.at<uchar>(0,0)=1;
- B.at<uchar>(0,1)=2;
- B.at<uchar>(0,2)=3;
- B.at<uchar>(1,0)=4;
- B.at<uchar>(1,1)=5;
- B.at<uchar>(1,2)=6;
- double AB=A.dot(B);
- cout<<"A=\n"<<A<<endl<<endl;
- cout<<"B=\n"<<B<<endl<<endl;
- cout<<"double类型的AB=\n"<<AB<<endl<<endl;
- system("pause");
- }
运行结果:
若对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声明:
- //! per-element matrix multiplication by means of matrix expressions
- MatExpr mul(InputArray m, double scale=1) const;
以简单的情况为例,对于2*2大小的Mat矩阵A和B:
对A和B执行mul运算:
mul说明:
1. mul操作不对参与运算的两个矩阵A、B有数据类型上的要求,但要求A,B类型一致,不然报错;
2. Mat AB=A.mul(B),若声明AB时没有定义AB的数据类型,则默认AB的数据类型跟A和B保存一致;
3. 若AB精度不够,可能产生溢出,溢出的值被置为当前精度下的最大值;
Opencv验证:
- #include "core/core.hpp"
- #include "iostream"
- using namespace std;
- using namespace cv;
- int main(int argc,char *argv[])
- {
- Mat A=Mat::ones(2,3,CV_8UC1);
- Mat B=Mat::ones(2,3,CV_8UC1);
- A.at<uchar>(0,0)=60;
- A.at<uchar>(0,1)=2;
- A.at<uchar>(0,2)=3;
- A.at<uchar>(1,0)=4;
- A.at<uchar>(1,1)=5;
- A.at<uchar>(1,2)=6;
- B.at<uchar>(0,0)=60;
- B.at<uchar>(0,1)=2;
- B.at<uchar>(0,2)=3;
- B.at<uchar>(1,0)=4;
- B.at<uchar>(1,1)=5;
- B.at<uchar>(1,2)=6;
- Mat AB=A.mul(B);
- cout<<"A=\n"<<A<<endl<<endl;
- cout<<"B=\n"<<B<<endl<<endl;
- cout<<"AB=\n"<<AB<<endl<<endl;
- system("pause");
- }
输出:
AB中第一个元素应该为60*60=360,但AB默认的类型为CV_8UC1,即最大值只能是255;所以执行mul运算一定要定义AB足够的精度,防止溢出。
Opencv中Mat矩阵相乘——点乘、dot、mul运算详解的更多相关文章
- opencv中mat矩阵如何debug
mat img 看type:img.type() rows:img.rows cols:img.cols channels:img.channels(),channels()这是一个函数,和上面两个有 ...
- OpenCV中Mat的列向量归一化
OpenCV中Mat的列向量归一化 http://blog.csdn.net/shaoxiaohu1/article/details/8287528 OpenCV中Mat的列向量归一化 标签: Ope ...
- opencv中mat类介绍
The class Mat represents an n-dimensional dense numerical single-channel or multi-channel array. It ...
- OpenCV中Mat的属性
OpenCV中Mat的属性 最近在做一OpenCV的图像轮廓检验,但当用到霍夫变换时才发现对Mat的属性了解不足.Mat在OpenCV中的地位是及其重要的,因此有必要做一个总结. 大体上来说,Mat是 ...
- OpenGL中的矩阵相乘
OpenGL中的矩阵相乘 1, 在OpenGL中所有的视图变换,模型变换 都是4×4矩阵,每个后续的glMultiMatrix*(N),或者变换函数,glTranslate* (),glRotate* ...
- OpenCV中Mat操作clone() 与copyto()的区别
OpenCV中Mat操作clone() 与copyto()的区别 // Mat is basically a class with two data parts: the matrix header ...
- OpenCV中Mat的基本用法:创建、复制
OpenCV中Mat的基本用法:创建.复制 一.Mat类的创建: 1.方法一: 通过读入一张图像,直接将其转换成Mat对象. Mat image = imread("test.jpg&quo ...
- PHP中IP地址与整型数字互相转换详解
这篇文章主要介绍了PHP中IP地址与整型数字互相转换详解,本文介绍了使用PHP函数ip2long与long2ip的使用,以及它们的BUG介绍,最后给出自己写的两个算法,需要的朋友可以参考下 IP转换成 ...
- ArcGIS中的北京54和西安80投影坐标系详解
ArcGIS中的北京54和西安80投影坐标系详解 1.首先理解地理坐标系(Geographic coordinate system),Geographic coordinate system直译为地理 ...
随机推荐
- PHP获取数组最后一个元素的键和值
<?php /** * PHP获取数组中最后一个元素下标和值 */ $arr = ['1' => 'name', '3' => 2, 5 => 6, 'name' => ...
- jquery----js/css 导入
<script type"text/javascript" src="JS文件"></script> <link rel = &q ...
- windows 系统常用操作
1.所有端口使用情况 netstat -ano 2.查询xxxx端口pid netstat -aon|findstr "xxxx" 3.根据端口Pid查详情 tasklist|fi ...
- C# mongoDB Driver 使用对象方式查询语法大全
#region 查询方法 /// <summary> /// 获取单个对象 /// </summary> /// <typeparam name="T" ...
- 安装ipython和jupyter
本节内容: 安装ipython 安装jupyter Pycharm介绍 Python软件包管理 一.安装ipython 1. python的交互式环境 2. 安装ipython 可以使用pip命令 ...
- Django ORM中使用update_or_create功能再解
以前,我解过这个问题,现在百度搜索,发了像也只能找到我这个帖子. https://www.cnblogs.com/aguncn/p/4922654.html 今天,看了看官方文档,关于这个update ...
- 扩展BSGS算法
求解A^x ≡ B mod P (P不一定是质数)的最小非负正整数解 先放几个同余定理: 一.判断如果B==1,那么x=0,算法结束 二.若gcd(A,P)不能整除 B,则 无解,算法结束 三.若gc ...
- python全栈开发day35-线程、协程
一.线程 1.线程 1).什么是线程 线程是cpu调度的最小单位 线程是进程的必要组成单位 一个进程里至少含有一个线程 2).主线程 程序开始运行的视乎,就产生了一个主线程来运行这个程序 3).子线程 ...
- Vijos1910 NOIP2014提高组 Day2T3 解方程 其他
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - Vijos1910 题意概括 已知多项式方程: a0+a1x+a2x2+...+anxn=0 求这个方程在[1 ...
- BZOJ4992 [Usaco2017 Feb]Why Did the Cow Cross the Road 最短路 SPFA
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ4992 题意概括 在一幅n*n的地图上,Amber从左上角走到右下角,每走一步需要花费时间t,每走完 ...