我对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
但是得到同样的错误。
在类似的说明中代码
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转换为某种静态类型
答案 0 :(得分:8)
失败与cython.boundscheck
无关。
Boundchecking只是检查您是否尝试访问不在那里的数组元素。例如,如果您有一个4号数组,并尝试访问元素5 - 使用boundscheck(True)
它会给您一个异常,boundscheck(False)
会导致未定义的行为(可能导致分段错误)。< / p>
编译失败的原因是另一个原因:您无法创建动态长度的静态数组!需要在编译时知道元素的数量,这只是c强制执行的(我猜)。
但是,您可以定义r
在compile 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)