7

以下

cimport cython

@cython.boundscheck(False)
def boundtest():
     cdef int r=4
     cdef double l[3]

工作正常。但是当我尝试这个时:

cimport cython

@cython.boundscheck(False)
def boundtest():
     cdef int r=4
     cdef double l[r]

我收到以下错误:

[1/1] Cythonizing test.pyx

Error compiling Cython file:
------------------------------------------------------------
...
cimport cython

@cython.boundscheck(False)
def boundtest():
     cdef int r=4
     cdef double l[r]
                   ^
------------------------------------------------------------

test.pyx:13:20: Not allowed in a constant expression
     

添加装饰器是因为找到了这个相关的 stackexchange 帖子并阅读了 Kurt W. Smith 的 Cython 书。据我所知,这应该可以告诉 Cython 不要担心可能由动态索引变量导致的越界错误,但由于某种原因它没有。我还尝试在编译器选项和全局范围内更改边界检查,但无济于事。

如果不是因为 Cython 文档声称是最新的,我会认为 boundscheck 已经贬值了。

4

1 回答 1

15

失败与否无关cython.boundscheck

边界检查只是检查您是否尝试访问不存在的数组元素。例如,如果您有一个大小为 4 的数组并尝试访问元素 5 -boundscheck(True)它会给您一个异常,boundscheck(False)它会导致未定义的行为(可能导致分段错误)。

编译失败的原因还有一个:不能创建动态长度的静态数组!在编译时需要知道元素的数量,这只是强制执行的(我猜)。

但是,您可以在编译时r定义为已知:

DEF r=4

cimport cython

@cython.boundscheck(False)
def boundtest():
    cdef double l[r]

但是,您可以简单地创建一个 NumPy 数组并将其存储在 memoryview 变量中:

cimport cython
import numpy as np

@cython.boundscheck(False)
def boundtest():
    cdef int r=4
    cdef double[:] l = np.empty(r, dtype=np.double)
于 2017-09-19T17:51:18.307 回答