4

我在 cython 中有一个 cdef 类,我想用 setattr 内置函数初始化它的字段。但是,当我这样做时,我得到了一个执行错误:

/path/.../cimul.cpython-34m.so in cimul.Simulation.__init__ (cimul.c:5100)()
AttributeError: 'Simulation' object has no attribute 'Re'

我的代码如下:

cdef class Simulation:
    cdef double Re, Pr, Ra, a, dt_security
    cdef int Nz, NFourier, freq_output, freq_critical_Ra, maxiter
    cdef bool verbose

    def __init__(self, *args, **kargs):
        param_list = {'Re': 1, 'Pr': 1, 'Ra': 1, 'a' : 1, 'Nz': 100,
                      'NFourier': 50, 'dt_security': 0.9,
                      'maxiter': 100, 'freq_output': 10,
                      'freq_critical_Ra':50, 'verbose': False}
        # save the default parameters
        for param, value in param_list.items():
            setattr(self, param, value)

你知道我该如何规避这个问题吗?

4

1 回答 1

2
  • 在类中定义(没有公共)属性时,您真正要做的是在 C 结构中定义一些字段。因此在编译(Cython + C)之后,属性的名称会丢失,它们仅由从 C 结构开头的一些偏移量标识。因此,它们不能从 Python 访问。

  • 如果添加cdef public,则 Cython 会添加一些属性访问函数,该函数不仅允许从 Python 进行访问,而且还保留了 C 结构中的关联标识符 <--> 偏移量。通过这些属性函数有一点开销。另请注意,这些函数执行 Python -> C 类型检查/转换。

现在要回答您的问题,您需要以某种方式保持关联标识 <--> 偏移。如果你想让事情变得更快,唯一的方法就是手工完成:

self.RE = param_list['RE']   # self.RE is a C struct access
self.Pr = param_list['Pr']
...
于 2014-06-02T20:35:50.940 回答