c++ - opencv C++ 神经网络 predict() 函数抛出 "Bad argument"错误

标签 c++ opencv neural-network

我已经成功地训练了一个神经网络来识别图像中的数字,并将网络参数保存到一个 .xml 文件中。

但是,当针对新图像测试网络时,代码在 predict() 阶段失败并出现错误:

OpenCV Error: Bad argument (Both input and output must be floating-point matrices of the same type and have the same number of rows) in CvANN_MLP::predict, file ........\opencv\modules\ml\src\ann_mlp.cpp, line 279.

ann_mlp.cpp 第 279 行是:

if( !CV_IS_MAT(_inputs) || !CV_IS_MAT(_outputs) ||
    !CV_ARE_TYPES_EQ(_inputs,_outputs) ||
    (CV_MAT_TYPE(_inputs->type) != CV_32FC1 &&
    CV_MAT_TYPE(_inputs->type) != CV_64FC1) ||
    _inputs->rows != _outputs->rows )
    CV_Error( CV_StsBadArg, "Both input and output must be floating-point matrices "
                            "of the same type and have the same number of rows" );

我通过运行这段代码检查了输入行:

cv::Size s = newVec.size();
    int rows = s.height;
    int cols = s.width;
    cout << "newVec dimensions: " << rows << " x " << cols << endl;

...结果是预期的 1 x 900 vector/矩阵。

我已经按照错误对话框将输入和输出矩阵设置为 CV_32FC1,如下所示:

输入矩阵

cv::Mat newVec(1, 900, CV_32FC1);
    newVec = crop_img.reshape(0, 1); //reshape / unroll image to vector
    CvMat n = newVec;
    newVec = cv::Mat(&n);

输出矩阵

    cv::Mat classOut = cvCreateMatHeader(1, CLASSES, CV_32FC1);

然后我尝试像这样运行预测函数:

CvANN_MLP* nnetwork = new CvANN_MLP;
nnetwork->load("nnetwork.xml", "nnetwork");

int maxIndex = 0;
cv::Mat classOut = cvCreateMatHeader(1, CLASSES, CV_32FC1);

//prediction
nnetwork->predict(newVec, classOut);
    float value;
    float maxValue = classOut.at<float>(0, 0);
    for (int index = 1; index<CLASSES; index++)
    {
        value = classOut.at<float>(0, index);
        if (value>maxValue)
        {
            maxValue = value;
            maxIndex = index;
        }
    }

有什么想法吗?非常感谢...

最佳答案

我怀疑问题出在您的输入上,而不是您的输出上。

首先,重要的是要了解 OpenCV 应该为此承担很多责任,而不是你。他们的 C++ API 非常平庸,给您带来了很大的困惑。

看,通常在 C++ 中,当您定义一个 1x900 的浮点矩阵时,它保持是一个浮点矩阵。 C++ 具有很强的类型安全性。

OpenCV 没有。如果将字节矩阵分配给浮点矩阵,后者将更改其类型(!)。 您的代码将 newVec 初始化为这样一个浮点矩阵,然后分配第二个矩阵,然后再分配另一个矩阵。我怀疑 crop_img 仍然是一个图像,即 8 位。 reshape 它会使它成为 1x900,但不是 float 。这就是 .convertTo 的工作。

关于c++ - opencv C++ 神经网络 predict() 函数抛出 "Bad argument"错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29374692/

相关文章:

c++ - 递归模板 : compilation error under g++

c++ - 为什么 boost filter_iterator 有奇怪的 make_filter_iterator 函数?

opencv - 裁剪瓶子的液体区域以进行处理

tensorflow - 你如何去规范化?

c++ - 表达式 :map/set iterator not incrementable

c++ - 是否有任何编译器或选项可以触发对无意义和错误的 switch 语句的警告?

opencv - glReadPixels 不适用于非方形宽度和高度?

java - 如何使用 OpenCV4 和 Android Studio(kotlin) 删除 Unresolved reference 错误

matlab - 在 GUI Matlab 中使用工作区变量

tensorflow - 手部地标坐标神经网络不收敛