6

所以,我有一个简单的事件库,用 C++ 编写并使用 Boost 库。我想把这个库暴露给 Python,所以我自然而然地转向了 Boost::Python。最终,我得到了要编译的代码,但现在我面临着一个相当大的问题:我的库使用了高阶编程技术。例如,该库由三个主要类组成:事件类、事件管理器类和事件侦听器类。事件侦听器类提出了问题。代码:

    class listener{
        public:
            listener(){}
            void alert(cham::event::event e){
                if (responses[e.getName()])
                    responses[e.getName()](e.getData());
            }
            void setResponse(std::string n, boost::function<void (std::string d)> c){responses.insert(make_pair(n, c));}
            void setManager(_manager<listener> *m){manager = m;}
        private:
            std::map<std::string, boost::function<void (std::string d)> > responses;
            _manager<listener> *manager;

如您所见,功能setResponse是问题所在。它需要一个函数传递给它,不幸的是,Boost::Python 在这种情况下没有应用它的转换器魔法。当调用如下:

>>> import chameleon
>>> man = chameleon.manager()
>>> lis = chameleon.listener()
>>> def oup(s):
...  print s
... 
>>> lis.setResponse("event", oup)

它给出了这个错误:

Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
Boost.Python.ArgumentError: Python argument types in
    listener.setResponse(listener, str, function)
did not match C++ signature:
    setResponse(cham::event::listener {lvalue}, std::string, boost::function<void ()(std::string)>)

所以,我的问题是,我该如何解决这个问题?它必须使用重载或包装器,因为我希望该库仍可被 C++ 调用。

4

1 回答 1

2

您将需要一个包装器setResponse,它采用 aboost::python::object而不是函数。它应该将其存储bp::object在已知位置(可能是子类的成员变量listener)。

然后将不同的 c++ 函数传递给 base setResponse,它将知道如何在bp::object. 如果要在不同的线程上调用事件,您还需要确保正确处理 python 的全局解释器锁,如下所述:boost.python 不支持并行性?.

于 2012-02-28T15:30:38.500 回答