如何解决Boost.Python 非常量与 const 方法参数
我一直试图让 Boost::Python 包装我的 C++ 代码。我想要的代码在下面并且已经简化。 BoundaryCheck 中的所有参数都不是常量,当我从 python 调用它时,出现错误:
Procedures.BoundaryCheck(Procedures,dict,str,list,int) 没有 匹配 C++ 签名
当我将原型和包装器更改为:
std::vector<std::vector<std::string> > BoundaryCheck(const std::map<std::string,std::string> &qcAttr,const std::string &targetSensor,const std::vector<std::vector<std::string> > &qcData,int verbose = 0) const;
.def("BoundaryCheck",(::std::vector<std::vector<std::string> > (::Products::Procedures::*)(::std::map< std::string,std::string > const &,std::string const &,std::vector<std::vector<std::string> > const &,int))(&::Products::Procedures::BoundaryCheck),BoundaryCheck_overloads(boost::python::args("qcAttr","targetSensor","qcData","verbose")));
事情按预期工作(我也更改了 Procedures.cpp,但应该很明显如何)。对我来说,为什么将方法参数更改为 const 会导致事情正常工作并不完全显而易见。除了 qcData 参数之外,将所有参数更改为 const 都可以,因为它由方法更改和返回。为什么 const 会导致这个工作?在不将 qcData 更改为 const 的情况下,我该怎么做才能使这项工作正常进行?
Procedures.h
#ifndef PROCEDURES_H_
#define PROCEDURES_H_
#include <string>
#include <vector>
#include <map>
namespace Products
{
class Procedures
{
public:
std::vector<std::vector<std::string> > BoundaryCheck(std::map<std::string,std::string &targetSensor,std::vector<std::vector<std::string> > &qcData,int verbose = 0);
}
}
#endif
Procedures.cpp
#include "Procedures.h"
#include <map>
#include <string>
#include <vector>
std::vector<std::vector<std::string> > Products::Procedures::BoundaryCheck(std::map<std::string,int verbose)
{
qcData[0][0] = "test";
return qcData;
}
wrapper.cpp
#include <boost/python.hpp>
#include <boost/python/overloads.hpp>
#include <boost/python/args.hpp>
#include "/data/alucard/include/Procedures.h"
BOOST_PYTHON_MEMBER_FUNCTION_OVERLOADS(BoundaryCheck_overloads,Products::Procedures::BoundaryCheck,3,4)
struct iterable_converter
{
template <typename Container> iterable_converter &from_python()
{
boost::python::converter::registry::push_back(&iterable_converter::convertible,&iterable_converter::construct<Container>,boost::python::type_id<Container>());
return *this;
}
static void* convertible(PyObject* object)
{
return PyObject_GetIter(object) ? object : NULL;
}
template <typename Container> static void construct(PyObject* object,boost::python::converter::rvalue_from_python_stage1_data* data)
{
std::vector< std::vector< std::string>> vecvec;
std::vector< std::string> vec;
PyObject *objectIN;
PyObject *value;
Py_ssize_t pos1 = PyList_Size(object);
boost::python::handle<> handle(boost::python::borrowed(object));
typedef boost::python::converter::rvalue_from_python_storage<Container> storage_type;
void* storage = reinterpret_cast<storage_type*>(data)->storage.bytes;
typedef boost::python::stl_input_iterator<typename Container::value_type> iterator;
new (storage) Container(iterator(boost::python::object(handle)),iterator());
data->convertible = storage;
}
};
struct pydict_converter
{
template <typename Container> pydict_converter &from_python()
{
boost::python::converter::registry::push_back(&pydict_converter::convertible,&pydict_converter::construct<Container>,boost::python::type_id<Container>());
return *this;
}
static void* convertible(PyObject* object)
{
return PyDict_Check(object) ? object : NULL;
}
template <typename Container> static void construct(PyObject* object,boost::python::converter::rvalue_from_python_stage1_data* data)
{
PyObject *key,*value;
Py_ssize_t pos = 0;
std::map<std::string,std::string> map;
boost::python::handle<> handle(boost::python::borrowed(object));
typedef boost::python::converter::rvalue_from_python_storage<Container> storage_type;
void* storage = reinterpret_cast<storage_type*>(data)->storage.bytes;
while (PyDict_Next(object,&pos,&key,&value))
{
char * k = PyUnicode_AsUTF8(key);
char * v = PyUnicode_AsUTF8(value);
map.insert(std::map<std::string,std::string>::value_type(k,v));
}
data->convertible = new (storage) Container(map);
}
};
struct vecvec_converter
{
static PyObject* convert(const std::vector< std::vector< std::string>> &vecvec)
{
PyObject* list = PyList_New(0);
PyObject* mval;
for (auto const& y : vecvec)
{
PyObject* templist = PyList_New(0);
for (auto const& x : y)
{
mval = PyBytes_FromString(&(x)[0]);
PyList_Append(templist,mval);
}
PyList_Append(list,templist);
}
return list;
}
};
BOOST_PYTHON_MODULE(procedures)
{
pydict_converter()
.from_python<std::map<std::string,std::string> >();
iterable_converter()
.from_python<std::vector<std::vector<std::string> > >();
iterable_converter()
.from_python<std::vector<std::string> >();
boost::python::to_python_converter<std::vector<std::map<std::string,std::string>>,vecmap_converter>();
//vector<vector> -> list(list)
boost::python::to_python_converter<std::vector<std::vector<std::string>>,vecvec_converter>();
boost::python::class_<Products::Procedures,boost::noncopyable>("Procedures")
.def("BoundaryCheck",std::string > &,std::string &,std::vector<std::vector<std::string> > &,"verbose")));
}
wrapper.py
import procedures
proc = procedures.Procedures()
qcAttr = {"m_sensorLoc_row":"0","m_qcboundstable":"qcbounds","m_sensortable":"sensor","m_unitLoc_row":"1","m_dataLoc_row":"2","m_flagtag":"_SCQC_Flag"}
qcData = [["Timestamp","AirTemp2m","AirTemp2m_SCQC_Flag"],["TS","C","Flag"],["2020-01-01 00:00:00+00","-100.99",""]]
data = proc.BoundaryCheck(qcAttr,qcData,1)
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。