我在pybind11中使用了不透明类型。例如,我为FromVec方法的参数类型stl容器std::vector<uint32_t>定义了一个
void FromVec(std::vector<uint32_t> vec);
PYBIND11_MAKE_OPAQUE(std::vector<uint32_t>);
PYBIND11_MODULE(tmap, m)
{
py::bind_vector<std::vector<uint32_t>>(m, "VectorUint",
"Unsigned 32-bit int vector.");
m.def("from_vec", &FromVec, py::arg("vec")
}有了这个,我现在可以(在Python中)执行以下操作:
vec = VectorUint([2, 66, 262, 662, 26, 62])
from_vec(vec)但是,类型转换不再起作用,因为该函数现在需要一个VectorUint,不再接受list,例如:
l = [2, 66, 262, 662, 26, 62]
from_vec(l)有没有一种方法可以同时支持间接类型和类型转换?也就是说,当传递list而不是间接绑定的VectorUint时,它会被转换为std::vector<uint32_t>,而不是Python抛出“不兼容的函数参数”?
发布于 2019-10-01 05:42:47
如果你需要从list自动转换成std::vector,你可以在你的绑定代码中添加这样的重载:
m.def("from_vec", &FromVec, py::arg("vec")); // accepts VectorUnit
m.def("from_vec", [](py::list& list){ /* do conversion manually and call FromVec */ } ); // accepts python list发布于 2020-03-19 01:02:09
只需在您的py::bind_vector后面添加以下内容
py::implicitly_convertible<py::list, std::vector<uint32_t>>();它甚至可以处理属性(例如用def_readwrite创建的属性)。您可以两全其美:向量是通过引用从python访问的,因此append、extend等可以按预期工作;您可以从python列表中赋值(当然会被复制)。
编辑:不幸的是,这个技巧不适用于std::map (不知道其他容器)。作为一种解决办法,您可以使用不同类型的中间映射,例如:
namespace pybind11::workaround
{
template<typename K, typename V>
class Map : public std::map<K, V> {};
}
namespace pybind11::detail
{
template<typename K, typename V>
struct type_caster<workaround::Map<K, V>> : map_caster<workaround::Map<K, V>, K, V> {};
}
inline void bindCommonTypes(pybind11::handle scope)
{
using IntMap = std::map<int, int>;
py::bind_map<IntMap>(scope, "IntMap")
.def(py::init([](const py::dict& dict) {
return dict.cast<pybind11::workaround::Map<int, int>();
}));
py::implicitly_convertible<py::dict, IntMap>();
}https://stackoverflow.com/questions/58169847
复制相似问题