gpt4 book ai didi

python - Boost.python自动转换参数

转载 作者:太空狗 更新时间:2023-10-29 21:21:49 27 4
gpt4 key购买 nike

我正在使用 boost.python 包装一个 C++ 类“A”,它以字符串作为构造函数。然后我有一个函数“fun(A& arg)”,它引用一个“A”作为参数。我想要一个用于“fun”的 python 包装器,如果我传递一个引用 python 字符串的变量,该引用首先会自动转换为对“A”的引用。

一个例子可能会有所帮助。在 python 方面,我希望能够做到这一点:

a = 'some string'
fun(a)

然后让“a”实际上是(对)一个“A”,而不是(对)原始字符串。我想这样做是因为我希望能够避免将其写成

a = A('some string')
fun(a)

(您可能有充分的理由怀疑这是一个相关的保护程序,但我们假设它对我很重要)。

这有可能吗?如果不使用 boost.python,也许直接使用 Python-C API?

注意:我知道如果我写

fun('some string')

无法将对字符串的引用转换为对某些其他类型的引用。

最佳答案

这是可能的,但解决方案可能取决于 Python 实现。

例如,在 Python 2.7 中,inspect模块和 sys.settrace()可用于修改locals()在特定的框架上。这也可以在 Python/C API 中完成,但在 Python 中操作 Python 框架通常更易于管理。

在下面的 example.py 中,update_locals() 函数将更新给定帧中的 locals():

import inspect
import sys


def _force_locals(old_frame, new_locals):
''' Force a new set of locals on a given frame.

:param old_frame: The frame to which locals will be applied.
:type old_frame: frame.
:param new_locals: What locals() should return for old_frame.
:type new_locals: dict.

.. note:: This function will force a custom trace function onto
the old_frame. Within the context of a trace function
(often used for debugging), a frame's f_locals is
modifiable. In most execution paths, f_locals is
writable but not modifiable.

'''
# Force tracing by setting the global tracing function to
# any non-None function.
if not sys.gettrace():
sys.settrace(lambda *args, **keys: None)

# Custom trace function that will force locals.
def trace(frame, event, arg):
# Update the frame's locals.
frame.f_locals.update(new_locals)
# Set frame to use default trace, preventing this trace
# function from resetting the locals on each trace call.
del frame.f_trace

# Set the old frame's trace function to the custom trace.
old_frame.f_trace = trace


def update_locals(frame, *refs):
''' Modifies a frame's locals based on the provided references.

:param frame: The frame from which a locals will be updated.
:type frame: frame.
:param refs: Iterable pair of (old_ref, new_ref) tuples.
:type refs: Iterable type of pairs.

'''
new_locals = frame.f_locals.copy()
has_changes = False

# If a frame's local has an identity patch with a provided
# reference, then update new_locals with the new reference.
for key, ref in new_locals.iteritems():
for old_ref, new_ref in refs:
if ref is old_ref:
new_locals[key] = new_ref
has_changes = True

# If new_locals has changes, then force them onto the frame.
if has_changes:
_force_locals(frame, new_locals)

交互使用:

>>> import example
>>> import inspect
>>> x = 42
>>> x
42
>>> example.update_locals(inspect.currentframe(), (x, '3.14'))
>>> x
'3.14'

x 变量引用了 int(42) 对象,但是 example.update_locals() 函数改变了 x 来引用 str('3.14') 对象。


如果参数是str 的实例,然后委托(delegate)给 C++ fun(A&) 函数并更新调用者的框架。

在下面的示例中,C++ Spam 类型和 fun(Spam&) 函数被暴露给 _example Python 模块。

#include <iostream>
#include <string>
#include <boost/python.hpp>

/// @brief Mockup type.
class Spam
{
public:
explicit Spam(std::string str)
: str_(str)
{}

void action()
{
std::cout << "Spam::action(): " << str_ << std::endl;
}

private:
std::string str_;
};

/// @brief Mockup function.
void fun(Spam& spam)
{
std::cout << "fun() -> ";
spam.action();
}

BOOST_PYTHON_MODULE(_example)
{
namespace python = boost::python;

python::class_<Spam>("Spam", python::init<std::string>());
python::def("fun", &fun);
}

更高级别的 example 模块将猴子补丁 _example.fun() 构造一个 Spam 对象,如果参数提供给 fun()str 的一个实例,并以与上述类似的方式操作调用者的框架:

from _example import *

import inspect
import sys


def _force_locals(old_frame, new_locals):
''' Force a new set of locals on a given frame.

:param old_frame: The frame to which locals will be applied.
:type old_frame: frame.
:param new_locals: What locals() should return for old_frame.
:type new_locals: dict.

.. note:: This function will force a custom trace function onto
the old_frame. Within the context of a trace function
(often used for debugging), a frame's f_locals is
modifiable. In most execution paths, f_locals is
writable but not modifiable.

'''
# Force tracing by setting the global tracing function to
# any non-None function.
if not sys.gettrace():
sys.settrace(lambda *args, **keys: None)

# Custom trace function that will force locals.
def trace(frame, event, arg):
# Update the frame's locals.
frame.f_locals.update(new_locals)
# Set frame to use default trace, preventing this trace
# function from resetting the locals on each trace call.
del frame.f_trace

# Set the old frame's trace function to the custom trace.
old_frame.f_trace = trace


def _update_locals(frame, *refs):
''' Modifies a frame's locals based on the provided references.

:param frame: The frame from which a locals will be updated.
:type frame: frame.
:param refs: Iterable pair of (old_ref, new_ref) tuples.
:type refs: Iterable type of pairs.

'''
new_locals = frame.f_locals.copy()
has_changes = False

# If a frame's local has an identity patch with a provided
# reference, then update new_locals with the new reference.
for key, ref in new_locals.iteritems():
for old_ref, new_ref in refs:
if ref is old_ref:
new_locals[key] = new_ref
has_changes = True

# If new_locals has changes, then force them onto the frame.
if has_changes:
_force_locals(frame, new_locals)


def _patch_fun():
old_fun = fun
# Create a function that will perform custom operations then
# delegate to the original function.
def patch(spam, *args, **kwargs):
if isinstance(spam, str):
old_spam, spam = spam, Spam(spam)

# In the caller's frame, force the variables that reference
# old_spam to now reference spam.
_update_locals(
inspect.currentframe(1), # Caller's frame.
(old_spam, spam))

return old_fun(spam, *args, **kwargs)
return patch

fun = _patch_fun()

交互使用:

>>> import example
>>> s1 = example.Spam('abc')
>>> type(s1)
<class '_example.Spam'>
>>> example.fun(s1)
fun() -> Spam::action(): abc
>>> type(s1) # s1's type has not changed.
<class '_example.Spam'>
>>> s2 = 'def'
>>> type(s2)
<type 'str'>
>>> example.fun(s2)
fun() -> Spam::action(): def
>>> type(s2) # s2's type has changed.
<class '_example.Spam'>
>>> example.fun('ghi')
fun() -> Spam::action(): ghi

请注意,上面示例中的帧操作仅修改了 fun() 的调用者的帧,而不是整个堆栈。

关于python - Boost.python自动转换参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21836508/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com