昊虹AI笔记网

 找回密码
 立即注册
搜索
查看: 1054|回复: 0
收起左侧

详解OpenCV的函数cv::add(),并附各种情况的示例代码和运行结果

[复制链接]

239

主题

241

帖子

931

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
931
昊虹君 发表于 2022-11-16 13:37 | 显示全部楼层 |阅读模式
详解OpenCV的函数cv::add(),并附各种情况的示例代码和运行结果

函数cv::add()用于实现两个Mat类矩阵相加,或者矩阵和标量相加。

函数add()的原型如下:
  1. void cv::add(InputArray src1,
  2.              InputArray src2,
  3.              OutputArray dst,
  4.              InputArray mask = noArray(),
  5.              int dtype = -1)
复制代码

参数意义如下:
src1---被加数;
src2---加数;
dst---运算结果;
mask---运算掩码;
dtype---输出矩阵的数据类型。

前四个参数没什么好说的,这里说下最后一个参数dtype,它用于设置输出矩阵的数据类型,具体情况如下:
The input arrays and the output array can all have the same or different depths.
For example, you can add a 16-bit unsigned array to a 8-bit signed array and store the sum as a 32-bit floating-point array. Depth of the output array is determined by the dtype parameter.
In the second and third cases above, as well as in the first case, when src1.depth() == src2.depth(), dtype can be set to the default -1. In this case, the output array will have the same depth as the input array, be it src1, src2 or both.

上面这段话简单明了,就不翻译了。需要说明的是,当两个相加的矩阵数据类型相同时,参数dtype可以用默认值-1。当两个相加的矩阵数据类型不一致时,是必须要设置输出矩阵的数据类型的,否则程序运行时会报错,比如下面的报错:
[Bash shell] 纯文本查看 复制代码
OpenCV Error: Bad argument (When the input arrays in add/subtract/multiply/divide functions have different types, the output array type must be explicitly specified) in cv::arithm_op, file C:\builds\master_PackSlave-win32-vc12-shared\opencv\modules\core\src\arithm.cpp, line 2011

上面报错的截图如下:


函数cv::add()的运算公式如下:


从上面的叙述可以看出:
1 函数add()可以进行掩码操作。

2 函数会对相加的结果进行饱和(saturate)操作,举个例子来说,如果数据类型为uhar,那么1+255=255,而不等于256。如果输出矩阵的数据类型为CV_32S,则不会进行饱和(saturate)操作,官方文档原话如下:
“Saturation is not applied when the output array has the depth CV_32S. You may even get result of an incorrect sign in the case of overflow.”

3 多通道的矩阵相加时,各个通道分别单独作相加算。

4 两个相同size和channels的矩阵与矩阵相加时(即上面的第一种情况)可以用操作符+代替,此时:
“cv::add(A1, B1, C1);”等效于“C1 = A1+B1;”
虽然可以这样等效操作,但是昊虹君还是建议诸位用标准函数式操作,这样能提高代码的健壮性。

接下来,上各种情况的示例代码:

目录
  • 示例代码1(相同数据类型的矩阵与矩阵相加)
  • 示例代码2(不同数据类型的矩阵与矩阵相加)
  • 示例代码3(矩阵与标量相加)
  • 示例代码4(带掩码操作的矩阵加法)

示例代码1(相同数据类型的矩阵与矩阵相加)
[C++] 纯文本查看 复制代码
//出处:昊虹AI笔记网(hhai.cc)
//用心记录计算机视觉和AI技术

//博主微信/QQ 2487872782
//QQ群 271891601
//欢迎技术交流与咨询

//OpenCV版本 OpenCV3.0

#include <opencv2/opencv.hpp>

#include <iostream>
using namespace std;


int main()
{

	cv::Mat A1 = (cv::Mat_<uchar>(2, 3) << 1, 2, 3, 4, 5, 6);
	cout << "A1中的数据为:\n" << A1 << endl << endl;


	cv::Mat B1 = (cv::Mat_<uchar>(2, 3) << 2, 3, 4, 5, 6, 250);
	cout << "B1中的数据为:\n" << B1 << endl << endl;


	//相同数据类型的矩阵与矩阵相加
	cv::Mat C1;
	cv::add(A1, B1, C1);
	cout << "C1中的数据为:\n" << C1 << endl << endl;

	return(0);
}

运行结果如下:


注意:C1中的最后一个元素因为饱和操作,值从256变成了255。

示例代码2(不同数据类型的矩阵与矩阵相加)
[C++] 纯文本查看 复制代码
//出处:昊虹AI笔记网(hhai.cc)
//用心记录计算机视觉和AI技术

//博主微信/QQ 2487872782
//QQ群 271891601
//欢迎技术交流与咨询

//OpenCV版本 OpenCV3.0

#include <opencv2/opencv.hpp>

#include <iostream>
using namespace std;

int main()
{

	cv::Mat A1 = (cv::Mat_<uchar>(2, 3) << 1, 2, 3, 4, 5, 6);
	cout << "A1中的数据为:\n" << A1 << endl << endl;


	cv::Mat B1 = (cv::Mat_<int>(2, 3) << 2, 3, 4, 5, 6, 250);
	cout << "B1中的数据为:\n" << B1 << endl << endl;


	//不同数据类型的矩阵与矩阵相加
	cv::Mat C1;
	cv::InputArray mask1 = cv::noArray();
	cv::add(A1, B1, C1, mask1, CV_16U);
	cout << "C1中的数据为:\n" << C1 << endl << endl;

	return(0);
}

运行结果如下:

注意,如果下面这条语句:
  1. cv::add(A1, B1, C1, mask1, CV_16U);
复制代码

如果写成了下面这样:
  1. cv::add(A1, B1, C1);
复制代码

则程序在运行时会因为A1与B1的数据类型不一样而报错,如下图所示:


示例代码3(矩阵与标量相加)
这里给一个单通道图像与标量相加的例子,
要注意,如果图像为多通道,
那么要求标量的数据类型为cv::Scalar,

关于cv::Scalar的详细介绍,
请参见博文:https://www.hhai.cc/thread-144-1-1.html

单通道图像与标量相加的示例代码如下:
[C++] 纯文本查看 复制代码
//出处:昊虹AI笔记网(hhai.cc)
//用心记录计算机视觉和AI技术

//博主微信/QQ 2487872782
//QQ群 271891601
//欢迎技术交流与咨询

//OpenCV版本 OpenCV3.0

#include <opencv2/opencv.hpp>

#include <iostream>
using namespace std;


int main()
{

	cv::Mat A1 = (cv::Mat_<uchar>(2, 3) << 1, 2, 3, 4, 5, 6);
	cout << "A1中的数据为:\n" << A1 << endl << endl;


	uchar b1 = 1;
	int b2 = 2;

	cv::Mat C1, C2;
	cv::InputArray mask1 = cv::noArray();
	cv::add(A1, b1, C1);//两个加数的数据类型相同
	cv::add(A1, b2, C2, mask1, CV_8U);//两个加数的数据类型不同
	cout << "C1中的数据为:\n" << C1 << endl << endl;
	cout << "C2中的数据为:\n" << C2 << endl << endl;


	return(0);
}

运行结果如下:

要注意:
此时当两个加数的数据类型不一致时,函数会自动处理,不会报错,但是为了程序的健壮性和可控性,昊虹君还是建议大家按上面的示例代码把结果数据类型控制好。
比如下面的例子:
[C++] 纯文本查看 复制代码
//OpenCV版本:3.0.0
//VS版本:2013

#include <opencv2/opencv.hpp>

#include <iostream>
using namespace std;


int main()
{

	cv::Mat A1 = (cv::Mat_<uchar>(2, 3) << 1, 2, 3, 4, 5, 6);
	cout << "A1中的数据为:\n" << A1 << endl << endl;

	uchar b1 = 1;
	double b2 = 2.3;

	cv::Mat C1, C2;
	cv::add(A1, b1, C1);
	cv::add(A1, b2, C2);
	cout << "C1中的数据为:\n" << C1 << endl << endl;
	cout << "C2中的数据为:\n" << C2 << endl << endl;


	return(0);
}

运行结果如下:

按照我们的思维习惯,输出矩阵C2中的数据应该都有小数位,但是事实上却没有,这就是函数自动处理的结果,有些自动处理并不是我们想要的结果,所以我们最好还是手动控制。

# 示例代码4(带掩码操作的矩阵加法)
[C++] 纯文本查看 复制代码
//出处:昊虹AI笔记网(hhai.cc)
//用心记录计算机视觉和AI技术

//博主微信/QQ 2487872782
//QQ群 271891601
//欢迎技术交流与咨询

//OpenCV版本 OpenCV3.0


#include <opencv2/opencv.hpp>

#include <iostream>
using namespace std;


int main()
{

	cv::Mat A1 = (cv::Mat_<uchar>(2, 3) << 1, 2, 3, 4, 5, 6);
	cout << "A1中的数据为:\n" << A1 << endl << endl;

	cv::Mat B1 = (cv::Mat_<uchar>(2, 3) << 2, 3, 4, 5, 6, 250);
	cout << "B1中的数据为:\n" << B1 << endl << endl;

	cv::Mat mask1(2, 3, CV_8UC1, cv::Scalar(1));
	mask1.at<uchar>(0, 1) = 0;

	cv::Mat C1;
	cv::add(A1, B1, C1, mask1);
	cout << "C1中的数据为:\n" << C1 << endl << endl;

	return(0);
}

运行结果如下:

上面的第0行,第1列的元素是没有被掩盖了加法操作的,所以其结果为0。
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|Archiver|昊虹AI笔记网 ( 蜀ICP备2024076726 )

GMT+8, 2024-5-17 01:32 , Processed in 0.026322 second(s), 23 queries .

Powered by Discuz! X3.4

Copyright © 2001-2021, Tencent Cloud.

快速回复 返回顶部 返回列表