为std :: string找到Boost Python No to_python转换器

DRa*_*ayX 17 c++ boost-python

所以,我正在尝试创建一个to_python转换器,它允许我从一个公开的函数返回一个boost :: optional,并且如果设置了可选项,则将其视为T,否则视为None.基于我在C++ Sig上发现的帖子,我编写了以下代码.

template<typename T>
struct optional_ : private boost::noncopyable {
  struct conversion {
    static PyObject* convert(boost::optional<T> const& value) {
      if (value) {
        return boost::python::to_python_value<T>()(*value);
      }
      Py_INCREF(Py_None);
      return Py_None;
    }
  };
  explicit optional_() {
    boost::python::to_python_converter<boost::optional<T>, conversion>();
  }
};
Run Code Online (Sandbox Code Playgroud)

据我所知,它适用于转换选项,但是python会抛出以下异常"TypeError:No to_python(by-value)convert for C++ type:std :: string".我知道C++能够将字符串转换为python,因为我的大多数公开函数都返回字符串.为什么boost :: python :: to_python_value不能识别它,我怎样才能利用它拥有的任何转换器?

通过更改为以下内容(基于本文)修复:

template<typename T>
struct optional_ : private boost::noncopyable {
  struct conversion {
    static PyObject* convert(boost::optional<T> const& value) {
      using namespace boost::python;
      return incref((value ? object(*value) : object()).ptr());
    }
  };
  explicit optional_() {
    boost::python::to_python_converter<boost::optional<T>, conversion>();
  }
};
Run Code Online (Sandbox Code Playgroud)

现在只做另一个版本,使其更干净,效果更好.

DRa*_*ayX 4

好的,这里是基于原始 C++ sig 帖子的整个可选转换器,但重写为使用高级 boost.python API(对奇怪的间距感到抱歉)。

template<typename T>
struct optional_ : private boost::noncopyable
{
  struct conversion :
    public boost::python::converter::expected_from_python_type<T>
  {
    static PyObject* convert(boost::optional<T> const& value) {
      using namespace boost::python;
      return incref((value ? object(*value) : object()).ptr());
    }
  };

  static void* convertible(PyObject *obj) {
    using namespace boost::python;
    return obj == Py_None || extract<T>(obj).check() ? obj : NULL;
  }

  static void constructor(PyObject *obj,
       boost::python::converter::rvalue_from_python_stage1_data *data)
  {
    using namespace boost::python;
    void *const storage =
      reinterpret_cast<
        converter::rvalue_from_python_storage<boost::optional<T> >*
      >(data)->storage.bytes;
    if(obj == Py_None) {
      new (storage) boost::optional<T>();
    } else {
      new (storage) boost::optional<T>(extract<T>(obj));
    }
    data->convertible = storage;
  }

  explicit optional_() {
    using namespace boost::python;
    if(!extract<boost::optional<T> >(object()).check()) {
      to_python_converter<boost::optional<T>, conversion, true>();
      converter::registry::push_back(
        &convertible,
        &constructor,
        type_id<boost::optional<T> >(),
        &conversion::get_pytype
      );
    }
  }
};
Run Code Online (Sandbox Code Playgroud)