- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我开始学习使用 boost python 并有一个菜鸟问题。
我想编写一个函数,可以将其参数的生命周期与其结果联系起来,这样当我调用 r = func(a)
时, 参数 a
如果我还有 r
的引用,就永远不会被销毁.文档建议使用 return_internal_reference
此类请求的调用策略。但这是否需要 r
成为a
的内部引用, 正如名字所暗示的那样?
在下面的(过度简化的)示例中,假设我想绑定(bind)输入数组 a
的生命周期使用生成的 lambda 函数,它不是输入的内部引用 a
.
#include <functional>
#include <boost/python.hpp>
#include <boost/python/return_internal_reference.hpp>
using namespace std;
using namespace boost::python;
function<float(int)> func(const float* a) {
return [=](int n) { return a[n]; };
}
BOOST_PYTHON_MODULE(test) {
def("func", func, return_internal_reference<1>());
}
我希望能够在python中做到以下几点:
f = func(a) # 'a' can be a temporary variable, say returned by another function
f(5) # but 'a' should not be destroyed at this step,
# because its lifetime is tied to 'f'
当我尝试编译上面的代码时,我遇到了下面列出的一大堆错误,但是如果我删除 return_internal_reference<1>()
调用策略,代码编译成功。
我很确定我错误地使用了这个调用策略,但我不确定如何正确使用它。任何指针将不胜感激。非常感谢!
$ g++ -std=c++11 -shared Test.cc -I/opt/local/Library/Frameworks/Python.framework/Versions/2.7/include/python2.7 -L/opt/local/lib -lboost_python-mt -lpython2.7 -o test.so
In file included from /opt/local/include/boost/preprocessor/iteration/detail/iter/forward1.hpp:52:0,
from /opt/local/include/boost/python/detail/invoke.hpp:63,
from /opt/local/include/boost/python/detail/caller.hpp:16,
from /opt/local/include/boost/python/object/function_handle.hpp:8,
from /opt/local/include/boost/python/converter/arg_to_python.hpp:19,
from /opt/local/include/boost/python/call.hpp:15,
from /opt/local/include/boost/python/object_core.hpp:14,
from /opt/local/include/boost/python/args.hpp:25,
from /opt/local/include/boost/python.hpp:11,
from Test.cc:2:
/opt/local/include/boost/python/detail/invoke.hpp: In instantiation of 'PyObject* boost::python::detail::invoke(boost::python::detail::invoke_tag_<false, false>, const RC&, F&, AC0&) [with RC = boost::python::detail::reference_existing_object_requires_a_pointer_or_reference_return_type<std::function<float(int)> >; F = std::function<float(int)> (*)(const float*); AC0 = boost::python::arg_from_python<const float*>; PyObject = _object]':
/opt/local/include/boost/python/detail/caller.hpp:223:13: required from 'PyObject* boost::python::detail::caller_arity<1u>::impl<F, Policies, Sig>::operator()(PyObject*, PyObject*) [with F = std::function<float(int)> (*)(const float*); Policies = boost::python::return_internal_reference<>; Sig = boost::mpl::vector2<std::function<float(int)>, const float*>; PyObject = _object]'
/opt/local/include/boost/python/object/py_function.hpp:38:33: required from 'PyObject* boost::python::objects::caller_py_function_impl<Caller>::operator()(PyObject*, PyObject*) [with Caller = boost::python::detail::caller<std::function<float(int)> (*)(const float*), boost::python::return_internal_reference<>, boost::mpl::vector2<std::function<float(int)>, const float*> >; PyObject = _object]'
Test.cc:14:1: required from here
/opt/local/include/boost/python/detail/invoke.hpp:75:82: error: no match for call to '(const boost::python::detail::reference_existing_object_requires_a_pointer_or_reference_return_type<std::function<float(int)> >) (std::function<float(int)>)'
return rc(f( BOOST_PP_ENUM_BINARY_PARAMS_Z(1, N, ac, () BOOST_PP_INTERCEPT) ));
^
In file included from /opt/local/include/boost/python/object/function_handle.hpp:8:0,
from /opt/local/include/boost/python/converter/arg_to_python.hpp:19,
from /opt/local/include/boost/python/call.hpp:15,
from /opt/local/include/boost/python/object_core.hpp:14,
from /opt/local/include/boost/python/args.hpp:25,
from /opt/local/include/boost/python.hpp:11,
from Test.cc:2:
/opt/local/include/boost/python/detail/caller.hpp: In instantiation of 'static const PyTypeObject* boost::python::detail::converter_target_type<ResultConverter>::get_pytype() [with ResultConverter = boost::python::detail::reference_existing_object_requires_a_pointer_or_reference_return_type<std::function<float(int)> >; PyTypeObject = _typeobject]':
/opt/local/include/boost/python/detail/caller.hpp:240:19: required from 'static boost::python::detail::py_func_sig_info boost::python::detail::caller_arity<1u>::impl<F, Policies, Sig>::signature() [with F = std::function<float(int)> (*)(const float*); Policies = boost::python::return_internal_reference<>; Sig = boost::mpl::vector2<std::function<float(int)>, const float*>]'
/opt/local/include/boost/python/object/py_function.hpp:48:35: required from 'boost::python::detail::py_func_sig_info boost::python::objects::caller_py_function_impl<Caller>::signature() const [with Caller = boost::python::detail::caller<std::function<float(int)> (*)(const float*), boost::python::return_internal_reference<>, boost::mpl::vector2<std::function<float(int)>, const float*> >]'
Test.cc:14:1: required from here
/opt/local/include/boost/python/detail/caller.hpp:102:109: error: 'struct boost::python::detail::reference_existing_object_requires_a_pointer_or_reference_return_type<std::function<float(int)> >' has no member named 'get_pytype'
return create_result_converter((PyObject*)0, (ResultConverter *)0, (ResultConverter *)0).get_pytype();
^
最佳答案
考虑使用 with_custodian_and_ward_postcall
CallPolicy .此策略允许按值返回返回类型,同时仍将另一个对象的生命周期延长到至少与返回对象的生命周期一样长。
BOOST_PYTHON_MODULE(test) {
def("func", func, with_custodian_and_ward_postcall<0, 1>());
}
如 return_internal_reference
中所述文档中,返回的对象引用了现有的内部对象:
return_internal_reference
[...] allow pointers and references to objects held internally [...] to be returned safely without making a copy of the referent.
该文档还简要提到了它对 with_custodian_and_ward_postcall
的使用。总之,return_internal_reference
对公开的函数有两个值得注意的影响:
owner_arg
指示的 ward 对象的生命周期,至少与保管人一样长。由于返回的 Python 对象指的是既没有显式所有权也没有共享所有权的现有对象,因此 Boost.Python 执行类型检查以防止创建悬空引用。在示例代码中,func()
按值返回一个仿函数,导致编译器错误提示返回类型必须是指针或引用:
struct boost::python::detail::reference_existing_object_requires_a_pointer_or_reference_return_type
To explicitly control the lifetime of the returned object, one should consider using return_value_policy
and models of ResultConverterGenerators. For example, if func()
returned a functor by pointer that was created by new()
, and wished to pass ownership of the object to Python while still maintaining the custodian and ward relationship, then one could chain policies with policy composition:
BOOST_PYTHON_MODULE(test) {
def("func", func,
return_value_policy<manage_new_object,
with_custodian_and_ward_postcall<0, 1> >());
}
这是一个完整的最小示例,它基于原始代码,详细输出到 demonstrate。 with_custodian_and_ward_postcall
行为:
#include <boost/python.hpp>
#include <iostream>
/// @brief Mockup class with verbose construction and destruction.
class foo
{
public:
foo() { std::cout << "foo() " << this << std::endl; }
foo(const foo&) { std::cout << "foo(const foo&) " << this << std::endl; }
~foo() { std::cout << "~foo() " << this << std::endl; }
};
/// @brief Mockup class with verbose construction and destruction.
class bar
{
public:
bar() { std::cout << "bar() " << this << std::endl; }
bar(const bar&) { std::cout << "bar(const bar&) " << this << std::endl; }
~bar() { std::cout << "~bar() " << this << std::endl; }
};
/// @brief Mockup factory function.
foo make_foo(bar& /* unused */)
{
return foo();
}
BOOST_PYTHON_MODULE(example)
{
namespace python = boost::python;
// Do not allow Foo to be explicitly created from its type.
python::class_<foo>("Foo", python::no_init);
python::class_<bar>("Bar", python::init<>());
// Expose make_foo, that returns a foo object when provided a
// bar object. The bar object's lifetime will be extended to
// be at least as long as that of the returned foo object.
python::def("make_foo", &make_foo,
python::with_custodian_and_ward_postcall<
0, // custodian = returned Foo object
1 // ward = provided Bar object
>());
}
交互使用:
>>> import example
>>> bar = example.Bar()
bar() 0x125ac30
>>> foo = example.make_foo(bar)
foo() 0x7fffa9b5efff
foo(const foo&) 0x7f1fcbe40090
~foo() 0x7fffa9b5efff
>>> bar = None
>>> foo = None
~foo() 0x7f1fcbe40090
~bar() 0x125ac30
请注意,尽管 bar
变量被设置为 None
,实际的 ward 对象仍然存在,直到 foo
保管人被销毁。
关于python - boost python : tie lifetime of argument to returned value using return_internal_reference,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26983556/
我正在尝试使用 flot 绘制 SQL 数据库中的数据图表,这是使用 php 收集的,然后使用 json 编码的。 目前看起来像: [{"month":"February","data":482},
我有一个来自 php 行的 json 结果,类似于 ["value"]["value"] 我尝试使用内爆函数,但得到的结果是“value”“value” |id_kategori|created_at
脚本 1 将记录 two 但浏览器仍会将 select 元素呈现为 One。该表单还将提交值 one。 脚本 2 将记录、呈现和提交 两个。我希望它们是同义词并做同样的事情。请解释它们为何不同,以及我
我的python字典结构是这样的: ips[host][ip] 每行 ips[host][ip] 看起来像这样: [host, ip, network, mask, broadcast, mac, g
在 C# 中 我正在关注的一本书对设置和获取属性提出了这样的建议: double pri_test; public double Test { get { return pri_test; }
您可能熟悉 enum 位掩码方案,例如: enum Flags { FLAG1 = 0x1, FLAG2 = 0x2, FLAG3 = 0x4, FLAG4 = 0x8
在一些地方我看到了(String)value。在一些地方value.toString() 这两者有什么区别,在什么情况下我需要使用哪一个。 new Long(value) 和 (Long)value
有没有什么时候 var result = !value ? null : value[0]; 不会等同于 var result = value ? value[0] : null; 最佳答案 在此处将
我正在使用扫描仪检测设备。目前,我的条形码的值为 2345345 A1。因此,当我扫描到记事本或文本编辑器时,输出将类似于 2345345 A1,这是正确的条形码值。 问题是: 当我第一次将条形码扫描
我正在读取 C# 中的资源文件并将其转换为 JSON 字符串格式。现在我想将该 JSON 字符串的值转换为键。 例子, [ { "key": "CreateAccount", "text":
我有以下问题: 我有一个数据框,最多可能有 600 万行左右。此数据框中的一列包含某些 ID。 ID NaN NaN D1 D1 D1 NaN D1 D1 NaN NaN NaN NaN D2 NaN
import java.util.*; import java.lang.*; class Main { public static void main (String[] args) thr
我目前正在开发我的应用程序,使其设计基于 Holo 主题。在全局范围内我想做的是工作,但我对文件夹 values、values-v11 和 values-v14. 所以我知道: values 的目标是
我遇到了一个非常奇怪的问题。 我的公司为我们的各种 Assets 使用集中式用户注册网络服务。我们一般通过HttpURLConnection使用请求方法GET向Web服务发送请求,通过qs设置参数。这
查询: UPDATE nominees SET votes = ( SELECT votes FROM nominees WHERE ID =1 ) +1 错误: You can't specify
如果我运行一段代码: obj = {}; obj['number'] = 1; obj['expressionS'] = 'Sin(0.5 * c1)'; obj['c
我正在为我的应用创建一个带有 Twitter 帐户的登录页面。当我构建我的项目时会发生上述错误。 values/strings.xml @dimen/abc_text_size_medium
我在搜索引擎中使用以下 View : CREATE VIEW msr_joined_view AS SELECT table1.id AS msr_id, table1.msr_number, tab
为什么验证会返回此错误。如何解决? ul#navigation li#navigation-3 a.current Value Error : background-position Too
我有一个数据名如下 import pandas as pd d = { 'Name' : ['James', 'John', 'Peter', 'Thomas', 'Jacob', 'Andr
我是一名优秀的程序员,十分优秀!