gpt4 book ai didi

c++ - 无法访问 IplImage 数据

转载 作者:太空宇宙 更新时间:2023-11-03 22:33:28 25 4
gpt4 key购买 nike

我一直在尝试比较 IplImage 对象中的数据内容。

我有以下内容:

   IplImage img1 = IplImage(cv::imread("C:\\TestIm\\barrier_snapshot1.png"),            
CV_LOAD_IMAGE_GRAYSCALE);
for (int i=0; i < img1.widthStep * img1.height; i++) {
cout << img1.imageData[i] << endl;
}

但是当我尝试打印它时,它导致了异常,我什至无法捕获它来打印消息并查看我做错了什么。我的图像是灰度图像,我相信如果我不使用 cvCreateImage() 就可以吗?我知道这将是一些愚蠢的或与数组访问相关的东西,我似乎无法从 IplImage 文档中轻松获得。

*为什么我要在我的设计中混合使用 C 和 C++ 代码? *

很遗憾,我别无选择!我正在从事一个改进了运动检测应用程序的项目。我的遗留应用程序源代码使用了大量的 BOOST 和 OpenCV 东西。特别是,它使用 IplImage*(我讨厌它,使生活变得困难并导致内存泄漏)来存储诸如图像 mask 之类的东西。我知道如果我长期保存 IplImage*,我将遇到非法引用和访问冲突。所以我保存了 IplImage* 指向的实际内容的拷贝。举个例子:

// getLongHistory() returns IplImage*
IplImage history_long = *(motionHistory.getLongHistory());

总共有 6 张使用 IplImage* 制作的蒙版图像。此刻,我在谴责决定在 IplImage* 中这样做的程序员。当我尝试加载这些蒙版图像时出现问题,我就是这样做的:

// Passing pointer to the address of the mask stored (alive in the memory)
motionHistory.setLongHistory(&(matcher.getCurrentSceneObject().getLongHistory()));

我相信我在处理 IplImage 对象的深拷贝和浅拷贝时遇到了问题。我相信将它从 IplImage* 保存为 cv::Mat 并从 cv::Mat 将其加载为 IplImage* 可能会减轻负担,因为我怀疑它可能在高级功能下做了SOMETHING 所以相应地完成数据和 ROI 的复制。但是,作为一个新手,我可以承担任何事情。请帮忙!

更新

在我的代码中,我过去这样做过:

/* I store all my mask images in a vector of pairs made of <int, IplImage>
* __MASK_LONG__ etc. are predefined intergers
* getMaskLong() etc. methods return IplImage* to the respective mask images.
*/

myImages.clear(); // To make sure that I have no extra stuff
myImages.push_back(std::make_pair<int, IplImage>(_MASK_LONG_, maskHistory.getMaskLong()));
myImages.push_back(std::make_pair<int, IplImage>(_MASK_SHORT_, maskHistory.getMaskShort()));

但是,在获得建议并进行了一些基本的 R&A 之后,我现在这样做是为了防止浅拷贝:

  myImages.clear(); 
myImages.push_back(std::make_pair<int, IplImage>(_MASK_LONG_, *cvCloneImage(maskHistory.getMaskLong())));
myImages.push_back(std::make_pair<int, IplImage>(_MASK_SHORT_, *cvCloneImage(maskHistory.getMaskShort())));

我可以确认这有效,因为我可以看到 OpenCV 窗口中加载了最新的蒙版图像!而且我非常确定在任何编程任务中至少进行 2/3 次深度复制是多么重要。所以谢谢你让我走上正轨。但是现在我遇到了在实现这些更改时想到的问题 - 内存分配失败。并且遇到了消息:

OpenCV Error: Insufficient memory (Failed to allocate 3686404 bytes) in OutOfMemoryError, 
file /home/naresh/OpenCV-2.4.0/modules/core/src/alloc.cpp, line 52

如果我足够深入地了解 C/C++,首先我犯下了将它们混合在一起的罪行(我别无选择!!这是一个遗留应用程序!)。其次,存在不匹配,即 alloc.cpp 文件(产生问题的地方)中对 malloc/free 的调用集不正确。或者可能只是堆已损坏或已满。我是不是傻了?

最佳答案

不要将 OpenCV 的 C 接口(interface)与 C++ 接口(interface)混用

理想情况下,您可以通过专门使用 C++ 接口(interface)来解决问题,如下所示:

cv::Mat gray = cv::imread("C:\\TestIm\\barrier_snapshot1.png", CV_LOAD_IMAGE_GRAYSCALE);

cv::Mat_<uchar>::iterator it = gray.begin<uchar>();
cv::Mat_<uchar>::iterator end = gray.end<uchar>();
for (; it != end; ++it)
{
cout << *it << endl;
}

或者:

cv::Mat gray = cv::imread("C:\\TestIm\\barrier_snapshot1.png", CV_LOAD_IMAGE_GRAYSCALE);

for (int i = 0; i < gray.cols; i++)
{
for (int j = 0; j < gray.rows; j++)
{
cout << gray[gray.cols * j + i] << endl;
}
}

是的,cv::imread()也可以将输入图像加载为灰度图像。但是,如果您真的需要坚持使用 C 接口(interface),则放弃 cv::imread() 并改用 cvLoadImage()。有几篇文章解释了如何执行此操作,请使用搜索框

如果您决定继续混合接口(interface)(请不要),check this thread因为它解释了如何将 IplImage* 转换为 cv::Mat

关于c++ - 无法访问 IplImage 数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14629719/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com