当你有个C++函数:
boost::any Get(std::string)
{
.....
}
void Set(std::string, boost::any)
{
.....
}
这样的两个API要传给python来执行的话,当你运行
a.set('hehe', 1);
返回的是类型不匹配。那么如何来修改boost.python,让它支持boost::any的格式呢,大家看过来:
首先打开你boost目录下的boost/python/converter/builtin_converters.hpp
在namespace converter里加上一句:
BOOST_PYTHON_DECL PyObject* do_return_to_python(boost::any);
然后加一句宏定义:
BOOST_PYTHON_TO_PYTHON_BY_value(boost::any, converter::do_return_to_python(x))
然后打开libs/python/src/converter/builtin_converters.cpp加上一段:
BOOST_PYTHON_DECL PyObject* do_return_to_python(boost::any x)
{
if(x.type() == typeid(int))
return PyInt_FromLong(any_cast<int>(x));
if(x.type() == typeid(float))
return PyFloat_FromDouble(any_cast<float>(x));
if(x.type() == typeid(double))
return PyFloat_FromDouble(any_cast<double>(x));
try{
std::string str = any_cast<std::string>(x);
return PyString_FromStringAndSize(str.data(),implicit_cast<int>(str.size()));
} catch (bad_any_cast &)
{
return PyInt_FromLong(0);
}
// todo, 新加对list,map,vector等的支持
}
ok, bjam一下,发现get好使了,呵呵。。然后来看set如何让它好使,set所需要的是python to c++的类型转换:
在libs/python/src/converter/builtin_converters.cpp里找到initialize_builtin_converters()这个函数,在里头加上一句:
slot_rvalue_from_python<boost::any, any_rvalue_from_python>();
然后在相同的文件里加上一个函数:
struct any_rvalue_from_python
{
static unaryfunc *get_slot(PyObject *obj)
{
if(PyInt_Check(obj) || PyString_Check(obj) || PyFloat_Check(obj))
return &py_object_identity;
return 0;
}
static boost::any extract(PyObject *obj)
{
boost::any res = boost::any();
if(PyInt_Check(obj))
{
int a = PyInt_AS_LONG(obj);
res = a;
}
if(PyFloat_Check(obj))
{
float f = (float)PyFloat_AS_DOUBLE(obj);
res = f;
}
if(PyString_Check(obj))
{
std::string str(PyString_AsString(obj),PyString_Size(obj));
res = str;
}
// todo, 新增对map,list,vector等的支持
return res;
}
};
OK,你会发现set也好使了。