我有一个c ++ boost python对象(PyObject *上的boost包装器),用于Python列表,
PyObject * pyList = func(...);
boost::python::object listObj(handle<>(boost::python::borrowed(pyList)));
我可以通过对其执行操作来验证这确实是一个列表,例如
boost::python::object np = import("numpy");
boost::python::np_difference = np.attr("diff");
np_difference(listObj);
for(int i=0; i<len(listObj); i++){
double value = boost::python::extract<double>(listObj[i]);
cout << i << " " << value << endl;
}
从i-1元素中减去第i个元素并创建一个新列表,然后将其中的每个元素提取并打印到使用C ++的stdout中。
我想要做的是将此listObj与我定义为
的打印功能一起使用boost::python::object print = std.attr("print");
但我只想打印我指定的其中一个元素。在python中我会写
print myList[0]
因为它只是一个python列表。但是当我尝试
时print(listObj[0]);
在c ++中使用boost python我得到了
Error in Python: <type 'exceptions.TypeError'>: No to_python (by-value) converter found for C++ type: boost::python::api::proxy<boost::python::api::item_policies>.
那么如何从python列表对象中访问单个元素并在python调用中使用它,如上面的print方法(或任何其他带有输入字符串的python函数)来自c ++?
答案 0 :(得分:4)
boost::python::object
的{{3}}会返回operator[]
个对象。虽然proxy
类隐式转换为boost::python::proxy
,但API中有许多区域需要显式转换。
明确构建boost::python::object
中的proxy
应解决转化异常:
print(boost::python::object(listObjString[0]));
这是一个完整的嵌入式Python示例,演示如何通过以下方式从列表中打印单个元素:
print
函数PyObject_Print()
函数__str__
提取对象的字符串表示并使用std::cout
#include <boost/foreach.hpp>
#include <boost/python.hpp>
#include <boost/range/irange.hpp>
/// @brief Default flag to have PyObject_Print use the object's
/// __str__ method. The python.h files only define the flag for
/// __repr__.
#define Py_PRINT_STR 0
int main()
{
Py_Initialize();
namespace python = boost::python;
try
{
// Create and populate a Python list.
// >>> list = [x for x in range(100, 103)]
python::list list;
BOOST_FOREACH(int x, boost::irange(100, 103))
list.append(x);
// The proxy returned from a Boost.Python's operator[] provides a
// user-defined conversion to a Boost.Python object. In most cases,
// explicitly invoking the conversion is required.
// Print list[0] using the built-in function print.
/// >>> getattr(__builtins__, 'print')(list[0])
python::object print =
python::import("__main__").attr("__builtins__").attr("print");
print(python::object(list[0]));
// Print list[1] using the Python/C API.
// >>> import sys; sys.stdout.write(list[1].__str__())
PyObject_Print(python::object(list[1]).ptr(), stdout, Py_PRINT_STR);
std::cout << std::endl;
// Print list[2] using the result of the object's __str__ method, and
// extract a C++ string.
std::cout << python::extract<std::string>(
python::object(list[2]).attr("__str__")())() << std::endl;
}
catch (python::error_already_set&)
{
PyErr_Print();
}
}
输出:
100
101
102
答案 1 :(得分:0)
修改强> Tanner Sansbury回答是解决这个问题的C ++方法,我在下面介绍的方法只是一个解决方法,它允许你将boost :: python命名空间中的python对象分配给__main__ python解释器命名空间。
知道了。灵感来自here,需要将boost :: python ::对象分配给python名称空间,即
boost::python::object main_module = boost::python::import("__main__");
boost::python::object main_namespace = main_module.attr("__dict__");
main_module.attr("myList") = listObj;
boost::python::exec("print myList", main_namespace);
希望这有帮助。