从boost :: python :: object列表中获取单个元素,以便在python例程中使用

时间:2014-03-26 22:56:22

标签: python c++ boost boost-python

我有一个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 ++?

2 个答案:

答案 0 :(得分:4)

boost::python::object的{​​{3}}会返回operator[]个对象。虽然proxy类隐式转换为boost::python::proxy,但API中有许多区域需要显式转换。

明确构建boost::python::object中的proxy应解决转化异常:

print(boost::python::object(listObjString[0]));

这是一个完整的嵌入式Python示例,演示如何通过以下方式从列表中打印单个元素:

  • Python内置print函数
  • Python / C API中的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);  

希望这有帮助。