在opencv中划分两个矩阵

Posted

技术标签:

【中文标题】在opencv中划分两个矩阵【英文标题】:divide two matrices in opencv 【发布时间】:2014-01-07 15:24:56 【问题描述】:

我想对两个 opencv CV_32S 矩阵(A 和 B)进行元素除法。

当 B 不为 0 时,我希望 C = A/B,否则为 0。

但我不确定是否理解 opencv 文档:

http://docs.opencv.org/modules/core/doc/operations_on_arrays.html#divide

上面写着:

当 src2(I) 为零时,dst(I) 也将为零。多通道数组的不同通道独立处理。

注意当输出数组的深度为 CV_32S 时,不应用饱和度。在溢出的情况下,您甚至可能得到错误符号的结果。

saturate() 函数是什么?我可以在 CV_32S 矩阵中安全地使用 divide(A,B,C) 吗? divide() 与 / 操作符有何不同?

===== 测试后编辑 =====

我的测试表明 / 运算符完全符合我的要求:当 B != 0 时 C = A/B,否则为 0。

【问题讨论】:

【参考方案1】:

saturate_cast 防止某些数据类型溢出,例如像素值200+200 减少到255 for CV_8U(否则会出现溢出并可能出现意外值)。

如果您想了解有关 saturate_cast 的更多信息,请查看该链接。 http://docs.opencv.org/modules/core/doc/intro.html#saturation-arithmetics

由于整数除法总是减少绝对值,所以整数除法应该不会发生溢出(或者我错过了什么?),所以我猜你应该是安全的。

【讨论】:

谢谢! divide() 与 / 运算符有何不同? ***:“数字 2,147,483,647 是计算中 32 位有符号整数的最大值”。因此,如果您将两个 32F 元素相除,如果结果是目标 32S 的溢出,则会发生意外行为。但是,对于输入 32S 矩阵,您是安全的。你需要重载 cv::Mat 的操作符/方法来实现额外的行为。 我不知道除法与标准 / 运算符有何不同。也许他们转换为浮点数并使用 SSE 指令,或者他们在内部使用 / 运算符,也许您可​​以在 openCV 源代码中找到这些部分;) 我查看了源代码(在matop.cpp第463行),但它充满了MatOp_Bin::makeExpr(),我不太明白......【参考方案2】:

所以,operator/cv::divide 实际上应该是相同的,只是运算符将返回一个矩阵表达式,其评估被延迟。最后,operator/ 也将调用cv::divide,可以看出 e。 G。对于一个简单的案例here。特别是结果应该相等。

不过,结果可能令人惊讶。使用两个整数矩阵进行除法,结果就像在浮点运算中完成,然后是round to nearest integer 更喜欢偶数(另请参阅nearbyint())。例如,使用两个 6x6 整数矩阵,您将得到

0/0 == 0    1/0 == 0    2/0 == 0    3/0 == 0    4/0 == 0    5/0 == 0
0/1 == 0    1/1 == 1    2/1 == 2    3/1 == 3    4/1 == 4    5/1 == 5
0/2 == 0    1/2 == 0    2/2 == 1    3/2 == 2    4/2 == 2    5/2 == 2
0/3 == 0    1/3 == 0    2/3 == 1    3/3 == 1    4/3 == 1    5/3 == 2
0/4 == 0    1/4 == 0    2/4 == 0    3/4 == 1    4/4 == 1    5/4 == 1
0/5 == 0    1/5 == 0    2/5 == 0    3/5 == 1    4/5 == 1    5/5 == 1

注意(如您在文档中所述),但还要注意(四舍五入)和 而(平局)。


注意: 两个浮点类型矩阵没有定义任何特殊行为,而是在被零除的情况下计算为 -inf、nan 或 inf。当您将两个浮点矩阵相除但请求整数结果时,这甚至成立。在这种情况下,-inf、nan 和 inf 将转换为最小值,这可能是一个错误(或者只是 OpenCV 没有定义)。


整数除法代码div.cpp(用g++ -std=c++11 div.cpp -o div -lopencv_core编译):

#include <opencv2/opencv.hpp>
#include <iostream>
#include <cstdint>

int main() 
    cv::Mat i1(6,6,CV_32S);
    cv::Mat i2(6,6,CV_32S);
    for (int y = 0; y < i1.rows; ++y) 
        for (int x = 0; x < i1.cols; ++x) 
            i1.at<int32_t>(y, x) = x;
            i2.at<int32_t>(y, x) = y;
        
    

    cv::Mat q;
    cv::divide(i1, i2, q);
//  q = i1 / i2;

    for (int y = 0; y < q.rows; ++y) 
        for (int x = 0; x < q.cols; ++x)
            std::cout << x << "/" << y << " == " << q.at<int32_t>(y, x) << "\t";
        std::cout << std::endl;
    

    return 0;

【讨论】:

以上是关于在opencv中划分两个矩阵的主要内容,如果未能解决你的问题,请参考以下文章

opencv中如何将两个类型为Mat的矩阵合为一个矩阵?

如何获取矩阵二范数 opencv

有没有办法防止opencv矩阵除法中的舍入

opencv六通道矩阵乘法

OpenCV 断言在矩阵乘法上失败

opencv中MatND和Mat的区别