gpt4 book ai didi

python - 如何从 Python/C 创建 lambda

转载 作者:太空宇宙 更新时间:2023-11-03 11:06:03 24 4
gpt4 key购买 nike

我们正在处理一些 Python/C-API 代码,我们遇到了一个想要传递回调的方法。该方法将定期更新回调作为反馈。事实证明,我们对定期反馈不那么感兴趣。禁用该方法的默认反馈机制的唯一方法是向其传递某种回调。

我们采用的技术是声明一个只返回 None 的模块级函数,即:

static PyObject*
donothing(PyObject* self, PyObject* args) {
return Py_None;
}

当然,这个函数也需要注册到modules方法表中,即:

static PyMethodDef methods[] = {
{"donothing", donothing, METH_VARARGS, "do nothing"},
...
{NULL}
};

然后,当我们去调用该方法时,我们需要获取对该方法的引用,即:PyObject_GetAttrString(module_reference, "donothing")。

所有这一切感觉就像我们花了太多时间转动我们的车轮而无所事事。然后我想到了......嘿,这似乎是 lambda x: None 的完美用法。但是在花了一个小时阅读 Python/C-API 文档之后,我无法弄清楚如何创建 lambda。

我看到页面上有对闭包的引用 http://docs.python.org/2/c-api/function.html ,但我无法理清有关如何创建它们的详细信息。

任何指针(或对 RTFM 的引用)将不胜感激。

最佳答案

lambda 表达式用于创建简单的匿名函数。这些有一个 PyFunction_Type包装 PyCode_Type 的对象,这是一大块可执行代码。但是你已经在 C 端了,所以创建一个 Python 函数会有点过分。相反,您应该创建一个 PyCFunction_Type 的对象.这类似于您尝试使用模块方法执行的操作。

C 中的样板文件也不会太大,但只有几行:

static PyObject *
donothing(PyObject *self, PyObject *args) {
Py_RETURN_NONE;
}
static PyMethodDef donothing_ml = {"donothing", donothing, METH_VARARGS, "doc"};

然后使用 PyCFunction_New(&donothing_ml, NULL) 创建对象产生 <built-in function donothing> .此功能独立于您的模块,可以像任何其他功能一样使用 PyObject .

这不是一个高水平lambda ,而是 lambda *args: None 的低级实现.


但是,如果您真的想创建一个高级别 lambda你可以用一个像dastrobu这样的语句来做到这一点提议

l = PyRun_String("lambda *args: None", Py_eval_input, PyEval_GetGlobals(), NULL);

如果你想自己组装,你可以做

PyCodeObject *c = (PyCodeObject *) Py_CompileString("None", "fn", Py_eval_input);
#if PY_MAJOR_VERSION >= 3
c->co_name = PyUnicode_FromString("<c-lambda>"); // function name
#else
c->co_name = PyString_FromString("<c-lambda>"); // function name
#endif
c->co_flags |= CO_VARARGS; // accept *args
c->co_nlocals = 1; // needed in Python 3
l = PyFunction_New((PyObject *) c, PyEval_GetGlobals());

在这两种情况下,您都会得到一个带有反汇编代码的函数 dis(l)相当于 lambda :

  1           0 LOAD_CONST               0 (None)
3 RETURN_VALUE

关于python - 如何从 Python/C 创建 lambda,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19549028/

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