2

我正在使用来自 OpenCV docs 的参考来创建自己的算法,该算法继承自 cv::Algorithm 。我已经创建了自己的类,这些类成功地从 cv::Algorithm 继承,但我在使用这个时遇到了困难,因为它有一个成员,该成员m_model是来自无法修改的库的成员,因为MyAlgorithm该类将功能包装在这个结构。

无论如何,我试图在结构中引用一个 `uchar[3] 的成员,所以我将它包装在 cv::Ptr 中。当我在 addParam 方法上编译没有 getter 或 setter 的程序时

obj.info()->addParam<uchar[3]>(obj, "arrPtr",  *arrPtr, false);

代码编译得很好,但是当我尝试将writeMyAlgorithm 对象归档时出现运行时错误,因为它无法获取数据。它正在寻找具有arr名称但不存在的成员变量。所以我为类成员中的arr参数定义了一些getter和setter方法。m_model

但是,我不确定如何将成员函数指针传递给 addParams 方法。我知道我不能像我目前在下面的代码中所做的那样将它们传递给 addParams 方法。我还尝试了以下方法:

obj.info()->addParam<uchar[3]>(obj, "arr",  *arrPtr, false, &MyAlgorithm::getArr, &MyAlgorithm::setArr);

但我得到一个编译错误:

cannot convert parameter 5 from 'cv::Ptr<_Tp> (__thiscall MyAlgorithm::* )(void)' to 'cv::Ptr<_Tp> (__thiscall cv::Algorithm::* )(void)'

下面是我的源代码的精简示例。任何帮助将不胜感激。

我的算法.h

class MyAlgorithm : public cv::Algorithm    {
public:
    //Other class logic
    cv::Ptr<uchar[3]> getArr();
    void setArr(const cv::Ptr<uchar[3]> &arrPtr);

    virtual cv::AlgorithmInfo* info() const;
protected:
    //define members such as m_model
};
cv::Algorithm* createMyAlgorithm();
cv::AlgorithmInfo& MyAlgorithm_info();

我的算法.cpp

cv::Algorithm* createMyAlgorithm()
{
   return new MyAlgorithm();
}

cv::AlgorithmInfo& MyAlgorithm_info() 
{
   static cv::AlgorithmInfo MyAlgorithm_info_var("MyAlgorithm", createMyAlgorithm);
   return MyAlgorithm_info_var;
}

cv::AlgorithmInfo* MyAlgorithm::info() const
{
   static volatile bool initialized = false;

   if( !initialized ) 
   { 
      initialized = true;
      MyAlgorithm obj; 
      cv::Ptr<uchar[3]> *arrPtr = new cv::Ptr<uchar[3]>(&(obj.m_model->arr));
      obj.info()->addParam<uchar[3]>(obj, "arr",  *arrPtr, false, &getArr, &setArr);
   } 
   return &MyAlgorithm_info();
}

cv::Ptr<uchar[3]> MyAlgorithm::getArr(){
   //Logic to get arr
}
void MyAlgorithm::setArr(const cv::Ptr<uchar[3]> &arrPtr){
   //Logic to set arr
}
4

1 回答 1

2

我设法让setterandgetter方法addParam起作用。问题是我需要static_cast像这样对父类执行:

typedef cv::Ptr<uchar[3]> (cv::Algorithm::*ArrGetter)();
typedef void (cv::Algorithm::*ArrSetter)(const cv::Ptr<uchar[3]> &);
obj.info()->addParam<uchar[3]>(obj, "arr",  *arrPtr, false, static_cast<ArrGetter>(&MyAlgorithm::getArr), static_cast<ArrSetter>(&MyAlgorithm::setArr));

但是,我还发现 OpenCV 不知道如何处理uchar[3]. 我尝试了一个cv::Vec3b似乎也不起作用的方法,所以我决定cv::Mat使用 setter 和 getter。所以最终的解决方案看起来像这样。

我的算法.h

class MyAlgorithm : public cv::Algorithm    {
public:
    typedef cv::Mat (cv::Algorithm::*ArrGetter)();
    typedef void (cv::Algorithm::*ArrSetter)(const cv::Mat &);

    //Other class logic
    cv::Mat getArr();
    void setArr(const cv::Mat &arrPtr);

    virtual cv::AlgorithmInfo* info() const;
protected:
    uchar[3] arr;
};
cv::Algorithm* createMyAlgorithm();
cv::AlgorithmInfo& MyAlgorithm_info();

我的算法.cpp

cv::AlgorithmInfo* MyAlgorithm::info() const
{
    static volatile bool initialized = false;

    if( !initialized ) 
    { 
        initialized = true;
        MyAlgorithm obj;
        cv::Vec3b arrVec(arr);
        obj.info()->addParam(obj, "arr",  (cv::Mat)arrVec, false, static_cast<ArrGetter>(&MyAlgorithm::getArr), static_cast<ArrSetter>(&MyAlgorithm::setArr));
    } 
    return &MyAlgorithm_info();
}

cv::Mat MyAlgorithm::getArr(){
    cv::Vec3b arrVec(arr);
    return (cv::Mat)arrVec;
}
void MyAlgorithm::setArr(const cv::Mat &arrMat){
    for (int i = 0; i < 3; i++)
        arr[i] = arrMat.at<uchar>(i);
}
于 2012-09-19T14:10:49.433 回答