我是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代码留在文件中,即使有编译错误?也许查看这个文件可以帮助我理解错误信息(即为什么这个变量被认为是本地的)......
答案 0 :(得分:3)
简短回答
使用global
语句在写入的每个函数中将poslast
显式声明为全局。
答案很长
Cython遵循Python绑定规则:如果在函数中为任何地方指定名称,则假定它是局部变量,除非您明确声明它是全局的。
我猜你在你的示例循环之后执行并赋值给poslast
,这使得它成为一个隐式声明的局部变量(object
类型)。然后循环似乎使用了这个尚未初始化的本地。
您的posarr_init
功能同样不正确。它分配隐式声明的本地poslast
,而不会触及全局。