- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
免责声明:是的,我知道 boost::python::map_indexing_suite
.
任务:我有一个 C++ 类,我想用 Boost.Python 包装它。它的构造函数采用 std::map
争论。这是 C++ header :
// myclass.hh
typedef std::map<int, float> mymap_t;
class MyClass {
public:
explicit MyClass(const mymap_t& m);
// ...
};
// ...
这是 Boost.Python 包装器(仅重要部分):
// myclasswrapper.cc
#include "mymap.hh"
#include "boost/python.hpp"
#include "boost/python/suite/indexing/map_indexing_suite.hpp"
namespace bpy = boost::python;
// wrapping mymap_t
bpy::class_<mymap_t>("MyMap")
.def(bpy::map_indexing_suite<mymap_t>())
;
// wrapping MyClass
bpy::class_<MyClass>("MyClass", "My example class",
bpy::init<mymap_t>() // ??? what to put here?
)
// .def(...method wrappers...)
;
这编译。但是,我无法创建映射 MyClass
来自 Python 端的对象,因为我不知道将什么作为参数传递给构造函数。字典不会转换为 std::map
-s 自动:
# python test
myclass = MyClass({1:3.14, 5:42.03})
口译员提示(理所当然):
Boost.Python.ArgumentError: Python argument types in
MyClass.__init__(MyClass, dict)
did not match C++ signature:
__init__(_object*, std::__1::map<int, float, ...
和MyMap
在 Python 端也不能用字典初始化。
在谷歌搜索了一天中最精彩的部分后,我只能找到采用 std::map
的“正常”方法的示例。用 .def(...)
映射的参数.而在 .def(...)
您不必明确指定映射方法的参数,它们会被神奇地发现。对于构造函数,您必须使用 boost::python::init<...>()
,或者至少这是我从文档中了解到的。
问题:
MyMap
中添加一些内容吗?包装帮助map_indexing_suite
从 Python 字典转换? boost::python::init<...>
中使用不同的模板参数吗?在MyClass
包装? 注意:我也看到了this accepted answer at SO ,然后我向下滚动并阅读@YvesgereY 的评论:
"For the record, map_indexing_suite solution doesn't work, since no implicit "dict->std::map" from_python converter will be applied."
然后我失去了信心:-)
最佳答案
我找到了一个很好的解决方案:添加了一个可以将 Python 字典转换为 std::map
的模板。 .逻辑基于this extremely useful primer , 稍作修改主要来自 this source file以及一些额外的评论。
下面是模板定义:
// dict2map.hh
#include "boost/python.hpp"
namespace bpy = boost::python;
/// This template encapsulates the conversion machinery.
template<typename key_t, typename val_t>
struct Dict2Map {
/// The type of the map we convert the Python dict into
typedef std::map<key_t, val_t> map_t;
/// constructor
/// registers the converter with the Boost.Python runtime
Dict2Map() {
bpy::converter::registry::push_back(
&convertible,
&construct,
bpy::type_id<map_t>()
#ifdef BOOST_PYTHON_SUPPORTS_PY_SIGNATURES
, &bpy::converter::wrap_pytype<&PyDict_Type>::get_pytype
#endif
);
}
/// Check if conversion is possible
static void* convertible(PyObject* objptr) {
return PyDict_Check(objptr)? objptr: nullptr;
}
/// Perform the conversion
static void construct(
PyObject* objptr,
bpy::converter::rvalue_from_python_stage1_data* data
) {
// convert the PyObject pointed to by `objptr` to a bpy::dict
bpy::handle<> objhandle{ bpy::borrowed(objptr) }; // "smart ptr"
bpy::dict d{ objhandle };
// get a pointer to memory into which we construct the map
// this is provided by the Python runtime
void* storage =
reinterpret_cast<
bpy::converter::rvalue_from_python_storage<map_t>*
>(data)->storage.bytes;
// placement-new allocate the result
new(storage) map_t{};
// iterate over the dictionary `d`, fill up the map `m`
map_t& m{ *(static_cast<map_t *>(storage)) };
bpy::list keys{ d.keys() };
int keycount{ static_cast<int>(bpy::len(keys)) };
for (int i = 0; i < keycount; ++i) {
// get the key
bpy::object keyobj{ keys[i] };
bpy::extract<key_t> keyproxy{ keyobj };
if (! keyproxy.check()) {
PyErr_SetString(PyExc_KeyError, "Bad key type");
bpy::throw_error_already_set();
}
key_t key = keyproxy();
// get the corresponding value
bpy::object valobj{ d[keyobj] };
bpy::extract<val_t> valproxy{ valobj };
if (! valproxy.check()) {
PyErr_SetString(PyExc_ValueError, "Bad value type");
bpy::throw_error_already_set();
}
val_t val = valproxy();
m[key] = val;
}
// remember the location for later
data->convertible = storage;
}
};
为了使用它,您必须创建一个 Dict2Map
实例,以便调用其构造函数。一种可能的方法是创建一个静态的 Dict2Map<key_t, val_t>
定义 Python 包装器的源文件中的变量。使用我的示例:
// myclasswrapper.cc
#include "mymap.hh"
#include "dict2map.hh"
// register the converter at runtime
static Dict2Map<char, double> reg{};
#include "boost/python.hpp" // not really necessary
namespace bpy = boost::python;
// wrapping MyClass
bpy::class_<MyClass>("MyClass", "My example class",
bpy::init<mymap_t>()
)
// .def(...method wrappers...)
;
现在可以创建MyClass
Python 端的对象是这样的:
myclass = MyClass({"foo":1, "bar":2})
编辑:Python 列表可以转换为 C++ std::vector
-s 以类似的方式。这是相应的模板:
template<typename elem_t>
struct List2Vec {
/// The type of the vector we convert the Python list into
typedef std::vector<elem_t> vec_t;
/// constructor
/// registers the converter
List2Vec() {
bpy::converter::registry::push_back(
&convertible,
&construct,
bpy::type_id<vec_t>()
#ifdef BOOST_PYTHON_SUPPORTS_PY_SIGNATURES
, &bpy::converter::wrap_pytype<&PyList_Type>::get_pytype
#endif
);
}
/// Check if conversion is possible
static void* convertible(PyObject* objptr) {
return PyList_Check(objptr)? objptr: nullptr;
}
/// Perform the conversion
static void construct(
PyObject* objptr,
bpy::converter::rvalue_from_python_stage1_data* data
) {
// convert the PyObject pointed to by `objptr` to a bpy::list
bpy::handle<> objhandle{ bpy::borrowed(objptr) }; // "smart ptr"
bpy::list lst{ objhandle };
// get a pointer to memory into which we construct the vector
// this is provided by the Python side somehow
void* storage =
reinterpret_cast<
bpy::converter::rvalue_from_python_storage<vec_t>*
>(data)->storage.bytes;
// placement-new allocate the result
new(storage) vec_t{};
// iterate over the list `lst`, fill up the vector `vec`
int elemcount{ static_cast<int>(bpy::len(lst)) };
vec_t& vec{ *(static_cast<vec_t *>(storage)) };
for (int i = 0; i < elemcount; ++i) {
// get the element
bpy::object elemobj{ lst[i] };
bpy::extract<elem_t> elemproxy{ elemobj };
if (! elemproxy.check()) {
PyErr_SetString(PyExc_ValueError, "Bad element type");
bpy::throw_error_already_set();
}
elem_t elem = elemproxy();
vec.push_back(elem);
}
// remember the location for later
data->convertible = storage;
}
};
关于python - 如何使用带有 Boost.Python 的 std::map 或 std::vector 参数的构造函数包装 C++ 类?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42952781/
简而言之:我想从可变参数模板参数中提取各种选项,但不仅通过标签而且通过那些参数的索引,这些参数是未知的 标签。我喜欢 boost 中的方法(例如 heap 或 lockfree 策略),但想让它与 S
我可以对单元格中的 excel IF 语句提供一些帮助吗? 它在做什么? 对“BaselineAmount”进行了哪些评估? =IF(BaselineAmount, (Variance/Baselin
我正在使用以下方法: public async Task Save(Foo foo,out int param) { ....... MySqlParameter prmparamID
我正在使用 CodeGear RAD Studio IDE。 为了使用命令行参数测试我的应用程序,我多次使用了“运行 -> 参数”菜单中的“参数”字段。 但是每次我给它提供一个新值时,它都无法从“下拉
我已经为信用卡类编写了一些代码,粘贴在下面。我有一个接受上述变量的构造函数,并且正在研究一些方法将这些变量格式化为字符串,以便最终输出将类似于 号码:1234 5678 9012 3456 截止日期:
MySql IN 参数 - 在存储过程中使用时,VarChar IN 参数 val 是否需要单引号? 我已经像平常一样创建了经典 ASP 代码,但我没有更新该列。 我需要引用 VarChar 参数吗?
给出了下面的开始,但似乎不知道如何完成它。本质上,如果我调用 myTest([one, Two, Three], 2); 它应该返回元素 third。必须使用for循环来找到我的解决方案。 funct
将 1113355579999 作为参数传递时,该值在函数内部变为 959050335。 调用(main.c): printf("%d\n", FindCommonDigit(111335557999
这个问题在这里已经有了答案: Is Java "pass-by-reference" or "pass-by-value"? (92 个回答) 关闭9年前。 public class StackOve
我真的很困惑,当像 1 == scanf("%lg", &entry) 交换为 scanf("%lg", &entry) == 1 没有区别。我的实验书上说的是前者,而我觉得后者是可以理解的。 1 =
我正在尝试使用调用 SetupDiGetDeviceRegistryProperty 的函数使用德尔福 7。该调用来自示例函数 SetupEnumAvailableComPorts .它看起来像这样:
我需要在现有项目上实现一些事件的显示。我无法更改数据库结构。 在我的 Controller 中,我(从 ajax 请求)传递了一个时间戳,并且我需要显示之前的 8 个事件。因此,如果时间戳是(转换后)
rails 新手。按照多态关联的教程,我遇到了这个以在create 和destroy 中设置@client。 @client = Client.find(params[:client_id] || p
通过将 VM 参数设置为 -Xmx1024m,我能够通过 Eclipse 运行 Java 程序-Xms256M。现在我想通过 Windows 中的 .bat 文件运行相同的 Java 程序 (jar)
我有一个 Delphi DLL,它在被 Delphi 应用程序调用时工作并导出声明为的方法: Procedure ProduceOutput(request,inputs:widestring; va
浏览完文档和示例后,我还没有弄清楚 schema.yaml 文件中的参数到底用在哪里。 在此处使用 AWS 代码示例:https://github.com/aws-samples/aws-proton
程序参数: procedure get_user_profile ( i_attuid in ras_user.attuid%type, i_data_group in data_g
我有一个字符串作为参数传递给我的存储过程。 dim AgentString as String = " 'test1', 'test2', 'test3' " 我想在 IN 中使用该参数声明。 AND
这个问题已经有答案了: When should I use "this" in a class? (17 个回答) 已关闭 6 年前。 我运行了一些java代码,我看到了一些我不太明白的东西。为什么下
我输入 scroll(0,10,200,10);但是当它运行时,它会传递字符串“xxpos”或“yypos”,我确实在没有撇号的情况下尝试过,但它就是行不通。 scroll = function(xp
我是一名优秀的程序员,十分优秀!