如何使用 pybind11 提供默认枚举值?

Baj*_*ile 3 c++ python enums default pybind11

我想要有接受枚举默认参数的函数。

但我无法在 PyBind11枚举示例和文档中找到提供枚举默认值的正确方法,例如:

struct Pet
{
    enum Kind
    {
        Dog = 0,
        Cat
    };

    Pet(const std::string &name) : name(name)
    {
    }

    void setName(const std::string &name_)
    {
        name = name_;
    }

    const std::string &getName() const
    {
        return name;
    }

    Kind test(Kind kind = Dog)
    {
        if(kind == Dog)
           std::cout << "Dog" << std::endl;

        if(kind == Cat)
           std::cout << "Cat" << std::endl;

        return kind;
    }

    std::string name;
};

PYBIND11_MODULE(pet,m)
{
    py::class_<Pet> pet(m, "Pet");
    pet.def(py::init<const std::string &>())
       .def("setName", &Pet::setName)
       .def("getName", &Pet::getName)
       .def("test", &Pet::test, py::arg("kind") = Pet::Kind::Dog)
       .def("__repr__", [](const Pet &a) { return "<example.Pet named '" + a.name + "'>"; }
    );

    py::enum_<Pet::Kind>(pet, "Kind")
        .value("Dog", Pet::Kind::Dog)
        .value("Cat", Pet::Kind::Cat)
        .export_values();
Run Code Online (Sandbox Code Playgroud)

但它不起作用:

py::arg("kind") = Pet::Kind::Dog 
Run Code Online (Sandbox Code Playgroud)

当我在 Python 中运行它时,我收到错误。

from pet import Pet as p
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
ImportError: arg(): could not convert default argument into a Python object (type not registered yet?). Compile in debug mode for more information.
Run Code Online (Sandbox Code Playgroud)

当尝试使用字符串值(即“Dog”或 1)初始化它时,我收到错误。

Wim*_*sen 5

这是一个简单的排序问题:加载时,执行模块定义中的语句来创建 Python 类、函数等。因此,Kind首先定义(即在 Python 端)以允许解释器在设置期间找到它test稍后定义时的默认值。即,使用这个命令:

PYBIND11_MODULE(pet,m)
{
    py::class_<Pet> pet(m, "Pet");

    py::enum_<Pet::Kind>(pet, "Kind")
        .value("Dog", Pet::Kind::Dog)
        .value("Cat", Pet::Kind::Cat)
        .export_values();

    pet.def(py::init<const std::string &>())
       .def("setName", &Pet::setName)
       .def("getName", &Pet::getName)
       .def("test", &Pet::test, py::arg("kind") = Pet::Kind::Dog)
       .def("__repr__", [](const Pet &a) { return "<example.Pet named '" + a.name + "'>"; }
    );

}
Run Code Online (Sandbox Code Playgroud)