【问题标题】:Passing a numpy function in python to C++ using SWIG directors使用 SWIG 控制器将 python 中的 numpy 函数传递给 C++
【发布时间】:2015-03-25 23:44:32
【问题描述】:

我的代码主要是 C++,但我想使用 SWIG 导向器选项在 python 中重新定义一个 C++ 虚函数,然后将该函数传递回 C++。下面是一个简化的例子。

这是我的 C++ 基类,带有一个虚函数 funfun 接受(double* x, int n),它允许我将一个 numpy 数组传递给它。

class base{
public:
    virtual double fun(double* x, int n){return 0.0;}
};

我有另一个类接受这个基类:

class main{
public:
    base* b_;

    main(base& b){
        b_ = &b; 
    }
};

这个类有一个构造函数,它接受一个基类并存储一个指向它的指针(b_)。

我可以成功地将 swig 接口编译成一个名为 mymodule 的模块,确保启用导向器(请参阅下面的 SWIG 接口文件),并且根据 SWIG 导向器文档,我继承了基类并重新定义了虚函数 @ 987654328@如下:

import mymodule
class base(mymodule.base):
    def __init__(self):
        super(base,self).__init__()
    def fun(self,x):
        return x[0]

然后我将它传递给main 类:

b = base()
m = mymodule.main(b)

但是,当我尝试m.b_.fun(array([1.,2.])) 时,我得到了TypeError

TypeError: 'SwigPyObject' object has no attribute '__getitem__'

关于如何让它发挥作用的任何想法?我认为当我将b 传递给mymodule.main 时,C++ 将x 视为双* 指针并且不知道如何处理它。

顺便说一下,这是我的 swig 接口文件:

%module(directors="1") mymodule

%{
#define SWIG_FILE_WITH_INIT
/* Includes the header in the wrapper code */
#include "myclasses.h"
%}

%feature("director") base;

%init %{
import_array();
%}

%apply (double* IN_ARRAY1, int DIM1) {(double* x, int n)};

/* Parse the header file to generate wrappers */
%include "myclasses.h"

【问题讨论】:

  • b.fun(array([1.,2.]) 工作吗?
  • 是的。但是当我将b 传递给m 时,它不会。
  • @kchow462 如果您想要任何不使用 swig 的工作解决方案,我已经使用 boost.python 从 C++ 到 python 再到 C++ 甚至 fork 再到 python 完成了类似的事情。查找文档非常乏味,但结果和生产力使我选择了这种方式。在我最近的工作中,django 从 C++ 启动了 zerorpc 接口并且工作正常。
  • 谢谢梅尔。我可能不得不这样做,因为 SWIG 似乎没有办法做到这一点,或者至少,我不知道有什么办法。

标签: python c++ arrays numpy swig


【解决方案1】:

这看起来像是来自 Python 的所有权问题。

向 SWIG 提供提示:

%new double *fun(double* IN_ARRAY1, int DIM1) {(double* x, int n)};

还要注意指向函数本身的指针。 当使用 %new 时,Python 模块将分配对象的正确所有权。这就是为什么您可以在调用中使用值但不能使用变量的原因。

【讨论】:

  • 我试过这个,但它似乎不起作用。我在我的 SWIG 接口文件中添加了 %new... 语句,但它仍然给我同样的错误。此外,它无法解决 C++ 函数接受 2 个参数而 Python 接受 1 个参数这一事实。
【解决方案2】:

director 似乎无法识别您的 fun(double* x, int n) 的 Python 实现;它没有做那个函数的重载。

我怀疑原因是您的 Python 实现使用 numpy 作为参数。请注意,director 将调用抽象函数或实现的函数。

另一方面,%apply (double* IN_ARRAY1, int DIM1) {(double* x, int n)} 仅声明用于在 Python 中包装函数的类型映射逻辑,而不是向主管声明。请注意,这就是 b.fun(array([1.,2.]) 起作用的原因!也就是说,你应该向导演声明类型映射逻辑。

您可以在这里找到操作方法:

%typemap(directorin,numinputs=1) (double *x, int n)
{
    npy_intp dim = $2;
    $input = PyArray_SimpleNewFromData(1, &dim, NPY_DOUBLE, (void *)$1);
}

希望这能解决您的问题和其他未来的问题:)。我遇到了类似的问题,花了几天时间试图弄清楚。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2019-01-28
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2016-09-15
    • 1970-01-01
    • 1970-01-01
    • 2010-09-27
    相关资源
    最近更新 更多