Cython编译错误 - 赋值前引用的局部变量

时间:2014-02-21 15:58:10

标签: python numpy compiler-errors cython

我是Cython的新手,并试图通过在关键AI模块中使用Cython代码来加速我的kivy手机游戏。我的代码如下:

import numpy as np
cimport numpy as np

 #not relevant parts

cdef np.ndarray posarr
cdef np.int poslast = 0
cdef np.int posidx = 0

def posarr_init(np.ndarray pawnpos, np.int act):
    poslast = 0
    # not relevant, but referencing poslast

 #not relevant, but including function where poslast is referenced (not assigned)

def consider_pawn(np.int x, np.int y):
    cdef np.int pact, posx, posy, resx, resy
    cdef np.int p
    cdef np.int found = 0
    #not relevant

    #in the code there is this line, posx and posy are local variables
                    posarr[poslast, posx, posy] = posarr[posidx, posx, posy]

Cython在编译过程中给出了这个错误:

Error compiling Cython file:

------------------------------------------------------------
...
                pact = 1
        if pact == 1:
            #pawn is active, create child position
            for posx in range(11):
                for posy in range(11):
                    posarr[poslast, posx, posy] = posarr[posidx, posx, posy]
                                 ^
------------------------------------------------------------

position.pyx:98:34: local variable 'poslast' referenced before assignment

我可以看到Cython编译错误的报告顺序与它们在代码中出现的顺序相同。我的问题是:

为什么Cython认为poslast是局部变量?

为什么它不认为它是以前函数中的局部变量?

生成的C文件为空,它只有一条消息,不应该在里面使用它。有没有办法强制Cython将C代码留在文件中,即使有编译错误?也许查看这个文件可以帮助我理解错误信息(即为什么这个变量被认为是本地的)......

1 个答案:

答案 0 :(得分:3)

简短回答

使用global语句在写入的每个函数中将poslast显式声明为全局。

答案很长

Cython遵循Python绑定规则:如果在函数中为任何地方指定名称,则假定它是局部变量,除非您明确声明它是全局的。

我猜你在你的示例循环之后执行并赋值给poslast,这使得它成为一个隐式声明的局部变量(object类型)。然后循环似乎使用了这个尚未初始化的本地。

您的posarr_init功能同样不正确。它分配隐式声明的本地poslast,而不会触及全局。