在OpenCV中移动图像内容

时间:2015-11-17 11:22:36

标签: c++ opencv opencv-mat

我的任务是以这种方式定期更新cv::Mat mr行和c列:

  1. 1列移动整个m向右移动,删除位置c-1的最后一列
  2. 0
  3. 位置随机生成新列
  4. 刷新m
  5. 的图

    为了清楚起见,这将导致一种带式输送机模拟。但是,当m必须转移时,问题出现在 1 点。

    我找到了两种不同的解决方案,即 A B ,两者都以相同的结果结束。这告诉我,我做错了。

    方法 A 如下:

    int offset = 1;
    cv::Mat tmp = cv::Mat::zeros(m.size(), m.type());
    cv::Rect rect_src(0, 0, m.cols-offset, m.rows);
    cv::Rect rect_dst(offset, 0, m.cols-offset, m.rows);
    
    cv::Mat in = m(rect_src);
    cv::Mat out = tmp(rect_dst);
    
    in.copyTo(out);
    m = temp;
    

    方法 B 如下:

    int offset = 1;
    cv::Mat trans_mat = (cv::Mat_<double>(2, 3) << 1, 0, offset, 0, 1, 0);
    cv::Mat warped;
    warpAffine(m, warped, trans_mat, m.size());
    m = warped;
    

    这里输出的例子是小m(随机值在左边产生):

    第1周期

    90      0       0       0       0       0       0       0       0
    143     0       0       0       0       0       0       0       0
    0       0       0       0       0       0       0       0       0
    

    第2周期

    0       0       90      0       0       0       0       0       0
    0       0       143     0       0       0       0       0       0
    0       0       0       0       0       0       0       0       0
    

    第3周期

    0       0       144     0       90      0       0       0       0
    0       0       161     0       143     0       0       0       0
    0       0       0       0       0       0       0       0       0
    

    很明显由零组成的额外列以某种方式出现 ...而且我真的无法弄清楚如何。

    P.S。如果我设置offset = 3,则输出会按因子2进行缩放,依此类推。

    90      0       0       0       0       0       0       0       0
    143     0       0       0       0       0       0       0       0
    0       0       0       0       0       0       0       0       0
    

    第2周期

    0       0       0       0       0       0       90      0       0
    0       0       0       0       0       0       143     0       0
    0       0       0       0       0       0       0       0       0
    

1 个答案:

答案 0 :(得分:2)

两种方法都可以正常工作,即使使用仿射变换在这里也是一种过度杀伤力。您可能在问题中没有显示的代码中出现错误。

此外,您可以使用colRange,这将简化您的代码。

检查两种方法的结果是否相同,并且不会出现额外的不需要的列:

#include <opencv2/opencv.hpp>
#include <iostream>
using namespace std;
using namespace cv;

int main()
{
    Mat1b img(3, 10);
    randu(img, Scalar(0), Scalar(255));

    Mat1b img2 = img.clone();

    //imshow("img", img);
    //waitKey();

    cout << img << endl << endl;

    int offset = 1;
    Mat trans_mat = (Mat_<double>(2, 3) << 1, 0, offset, 0, 1, 0);

    for (int i = 0; i < 100; ++i)
    {
        // Random data
        Mat1b randomData(img.rows, offset);
        randu(randomData, Scalar(0), Scalar(255));

        // Copying roi
        img.colRange(0, img.cols - offset).copyTo(img.colRange(offset, img.cols));
        randomData.copyTo(img.colRange(0, offset));
        //randu(img.colRange(0, offset), Scalar(0), Scalar(255));

        // Warping
        cv::Mat warped;
        warpAffine(img2, warped, trans_mat, img2.size());

        img2 = warped.clone();
        randomData.copyTo(img2.colRange(0, offset));
        //randu(img2.colRange(0, offset), Scalar(0), Scalar(255));

        //imshow("img", img2);
        //waitKey();

        cout << img << endl << endl;
        cout << img2 << endl << endl;

    }   
    return 0;
}

这是第一次迭代的数据。

原始数据

[ 91,   2,  79, 179,  52, 205, 236,   8, 181, 239;
  26, 248, 207, 218,  45, 183, 158, 101, 102,  18;
 118,  68, 210, 139, 198, 207, 211, 181, 162, 197]

通过复制roi来移动数据

[191,  91,   2,  79, 179,  52, 205, 236,   8, 181;
 196,  26, 248, 207, 218,  45, 183, 158, 101, 102;
  40, 118,  68, 210, 139, 198, 207, 211, 181, 162]

数据因翘曲而改变

[191,  91,   2,  79, 179,  52, 205, 236,   8, 181;
 196,  26, 248, 207, 218,  45, 183, 158, 101, 102;
  40, 118,  68, 210, 139, 198, 207, 211, 181, 162]