我试图获取函数的源代码,向其中添加代码,然后将其放回原始函数中.
基本上是这样的:
new_code = change_code(original_code)
throwaway_module = ModuleType('m')
exec(new_code, throwaway_module.__dict__)
func.__code__ = getattr(throwaway_module, func.__name__).__code__
当new_code不包含任何不在原始函数中的名称时,这非常有效.
但是当new_code包含原始func中没有的变量名时,那么在最后一行我得到以下错误:
ValueError: func() requires a code object with 1 free vars, not 0
有任何想法吗?
编辑:
似乎我已经在CPython源代码中找到了引发此异常的位置(文件funcobject.c).为清晰起见省略了一些线条:
static int
func_set_code(PyFunctionObject *op, PyObject *value, void *Py_UNUSED(ignored))
{
Py_ssize_t nfree, nclosure;
// ... lines omitted
nfree = PyCode_GetNumFree((PyCodeObject *)value);
nclosure = (op->func_closure == NULL ? 0 :
PyTuple_GET_SIZE(op->func_closure));
if (nclosure != nfree) {
PyErr_Format(PyExc_ValueError,
"%U() requires a code object with %zd free vars,"
" not %zd",
op->func_name,
nclosure, nfree);
return -1;
}
Py_INCREF(value);
Py_XSETREF(op->func_code, value);
return 0;
}
这有助于你帮助我吗?
标签:python,cpython,metaprogramming