3

首先我必须说,作为一名 Python 程序员,我可能从错误的角度看待这个问题,自从我在大学写最后一个 c++ 代码以来已经过去了很多年。

我遇到了一些问题,尝试使用 firebreath 创建一个混合 python/c++ 插件。到目前为止,我已经成功地使用 boost/python.h 集成了所有部分,但是当我尝试从 python 中触发事件时出现了问题。我偶然发现了必须将 python 函数与 c++ 函数绑定在一起的问题(使用 BOOST_PYTHON_MODULE)。首先,我尝试直接将 python 与我的 JSAPI 派生类 fbtestconpythonAPI 绑定,这种方法的问题似乎是缺少对浏览器实例化的 JSAPI 对象的引用,这给了我在 python 函数和 c++ 等效项之间的各种签名不匹配问题执行时间处理时间。

我唯一要解决这个问题(我同意,这是一个丑陋的肮脏解决方案),是使用我用 set_pluginPointer 手动初始化的全局指针。到目前为止,这实际上效果很好,但我知道这不是正确的方法。我读到我不应该使用带有 JSAPI 对象的“原始”指针,但我不确定如何用 shared_ptr 替换它来实现这个特定的实现。另一个问题是全局变量,它在所有实例之间共享,例如,导致所有事件在最后打开的选项卡/窗口上触发。解决后者的一种方法是创建某种数组,其索引为当前窗口/线程 ID,这是我应该能够从我的 JSAPI 对象和 python/c++ 函数访问的东西。

当然,我很开放,并且会非常感谢任何建议,关于如何改进/修复这个特定的解决方法,或者更好地,在没有黑客攻击的情况下交流 boost::python 和 firebreath 的正确方法。

下面是插件代码的相关部分

// Global pointer to plugin instance
fbtestconpythonAPI *fbtestPtr;
void fbtestconpythonAPI::set_pluginPointer(const std::string& val){
    m_testString = val;
    fbtestPtr = this; //Global pointer initialization
}

void echo(std::string x){
    // Firing the echo event on the plugin instance using the global raw pointer
    fbtestPtr->fire_echo(x, 1);
}

BOOST_PYTHON_MODULE(Pointless) {
    def("echo", echo);
}

FB::variant fbtestconpythonAPI::echo(const FB::variant& msg){
    int result_value;
    Py_Initialize(); 

    try {
        initPointless(); // initialize Pointless

        PyRun_SimpleString("import Pointless");
        PyRun_SimpleString("Pointless.echo('hello world')");

        object module(handle<>(borrowed(PyImport_AddModule("__main__"))));
        object dictionary = module.attr("__dict__");
    } catch (error_already_set) {
        PyErr_Print();
    }

    Py_Finalize();
    return 0;
}
4

1 回答 1

1

快速浏览一下,您将像这样设置您的导出类:

class_<YourAPI, boost::noncopyable>("YourAPI", no_init)
    .def("function", &YourAPI::function);

然后,您可以将对您的 C++ 实例的引用传递给 Python,从而允许您在其上调用函数,从而触发事件。

于 2012-02-17T00:58:44.890 回答