Cython"在常量表达式中不允许",boundscheck False不起作用

时间:2017-09-19 17:41:54

标签: python python-3.x cython cythonize

我对Cython比较陌生,并且遇到了我的研究失败的错误(我在spyder中使用Python3而我的Sython版本是0.26)

我试过了:

import cython

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

它工作正常。但后来我尝试了这个:

import cython

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

我收到错误

[1/1] Cythonizing test.pyx

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

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

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

由于找到了this相关的stackexchange帖子并阅读了Kurt W. Smith撰写的Cython书籍,因此添加了装饰器。据我所知,这应该告诉Cython不要担心可能因动态索引变量而导致的超出界限错误,但由于某种原因它不会。我也尝试在编译器选项中更改boundscheck并在全局范围内无效。

如果Cython文档没有声称是最新的,我会认为boundscheck已被折旧。

更新

我意识到我使用了import Cython而不是cimport cython。我再次尝试

cimport cython

但是得到同样的错误。

更新2

在类似的说明中代码

cdef int N = 3
cdef double[:] lout = array.array('d', N)

抛出错误

TypeError: 'int' object is not iterable

我假设出于同样的原因,C无法处理(可能)动态数组分配。相反,我们必须使用

cdef double[:] lout = numpy.empty(N, 'd')

我假设有一条线在将N放入C阵列之前将N转换为某种静态类型

1 个答案:

答案 0 :(得分:8)

失败与cython.boundscheck无关。

Boundchecking只是检查您是否尝试访问不在那里的数组元素。例如,如果您有一个4号数组,并尝试访问元素5 - 使用boundscheck(True)它会给您一个异常,boundscheck(False)会导致未定义的行为(可能导致分段错误)。< / p>

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

但是,您可以定义rcompile time知道:

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)