3

我想用 PyBind11 绑定一个 C++ 函数。问题是这个函数有一个带有双指针的参数,编译器会引发错误

error: cannot initialize a parameter of type 'char **' with an rvalue of type 'typename make_caster<char **>::cast_op_type<typename std::add_rvalue_reference<char**>::type>' (aka 'char *').

具体来说,代码如下所示:

#include <pybind11/pybind11.h>

#include <iostream>

namespace py = pybind11;

void parse_args(int argn__, char** argv__)
{
  for(int i = 1; i < argn__; ++i)
    {
      std::cout<< argv__[i];
    }
}

PYBIND11_MODULE(argv_bind, m) {
   m.def("parse_args", &parse_args);     
}
4

1 回答 1

7

对于非简单参数类型(例如您的char **),您必须包装函数本身,您的包装器提供从更简单类型或 Python 对象到所需类型的逻辑。

在这种情况下,char *实际上是一个非常不安全的类型,因为它没有进行内存清理;为了使这项工作稳健地进行,您需要获取std::strings,然后您可以从中访问char *viac_str()方法。(在 pybind11 中,即使您使用char *参数,它们实际上也只是指向std::string参数的指针:Python API 故意不让调用者访问 Python 字符串的内部存储)。

因此,您只需要将它与本身可以将整个事物公开为 a 的类型结合起来char **,这几乎std::vector就是设计要做的。这是一个适合您的示例代码的包装器:

#include <pybind11/stl.h>

m.def("parse_args", [](std::vector<std::string> args) {
    std::vector<char *> cstrs;
    cstrs.reserve(args.size());
    for (auto &s : args) cstrs.push_back(const_cast<char *>(s.c_str()));
    return parse_args(cstrs.size(), cstrs.data());
});

(如果你parse_args真的需要 aconst char **你可以放弃const_cast)。

于 2018-03-24T23:48:06.350 回答