如何在 Cython 中使用 atexit?

How to use atexit in Cython?

我正在尝试在 cython 中编译以下代码

from cython_gsl cimport *

import atexit

cdef gsl_rng_type * rng_T = gsl_rng_default
cdef gsl_rng * rng_r

gsl_rng_env_setup()
rng_r = gsl_rng_alloc(rng_T)

@atexit.register
def free_gsl_rng():
    gsl_rng_free(rng_r)

但我总是得到错误

赋值给非左值 'atexit'

对应的.pxd文件

from cython_gsl cimport *

cdef gsl_rng * rng_r

我实际上是用 SageMath 8.7 编译的

sage setup.py build_ext --inplace

这是我的 setup.py

from distutils.core import setup
from distutils.extension import Extension
from Cython.Distutils import build_ext
from Cython.Build import cythonize
import cython_gsl

extensions = [
        Extension("gsl_rand", ["gsl_rand.pyx"],
            libraries=cython_gsl.get_libraries(),
            library_dirs=[cython_gsl.get_library_dir()],
            include_dirs=[cython_gsl.get_cython_include_dir()]
            ),
        ]

setup(
        name='Simulation of k-cut on conditional Galton-Watson trees',
        cmdclass={'build_ext': build_ext},
        include_dirs = [cython_gsl.get_include()],
        ext_modules=cythonize(extensions),
        )

Full error log here

sage setup.py build_ext --inplace
Compiling gsl_rand.pyx because it changed.
[1/1] Cythonizing gsl_rand.pyx
/home/xing/Downloads/software/sage/8.7/local/lib/python2.7/site-packages/Cython/Compiler/Main.py:367: FutureWarning: Cython directive 'language_level' not set, using 2 for now (Py2). This will change in a later release! File: /home/xing/Dropbox/Research/2017/k-cut/GW/sage/moments/cython-v5/gsl_rand.pxd
  tree = Parsing.p_module(s, pxd, full_module_name)
warning: /home/xing/Downloads/software/sage/8.7/local/lib/python2.7/site-packages/cython_gsl/gsl_integration.pxd:65:9: 'GSL_EMAXITER' redeclared 
warning: /home/xing/Downloads/software/sage/8.7/local/lib/python2.7/site-packages/cython_gsl/gsl_integration.pxd:67:9: 'GSL_EROUND' redeclared 
warning: /home/xing/Downloads/software/sage/8.7/local/lib/python2.7/site-packages/cython_gsl/gsl_integration.pxd:69:9: 'GSL_ESING' redeclared 
warning: /home/xing/Downloads/software/sage/8.7/local/lib/python2.7/site-packages/cython_gsl/gsl_integration.pxd:71:9: 'GSL_EDIVERGE' redeclared 
warning: gsl_rand.pyx:14:0: Overriding cdef method with def method.

Error compiling Cython file:
------------------------------------------------------------
...
# cython: profile=False

from cython_gsl cimport *

import atexit
      ^
------------------------------------------------------------

gsl_rand.pyx:5:7: Assignment to non-lvalue 'atexit'

Error compiling Cython file:
------------------------------------------------------------
...
cdef gsl_rng * rng_r

gsl_rng_env_setup()
rng_r = gsl_rng_alloc(rng_T)

@atexit.register
^
------------------------------------------------------------

gsl_rand.pyx:13:0: Object of type 'int (void (*)(void) nogil) nogil' has no attribute 'register'
Traceback (most recent call last):
  File "setup.py", line 29, in <module>
    ext_modules=cythonize(extensions),
  File "/home/xing/Downloads/software/sage/8.7/local/lib/python2.7/site-packages/Cython/Build/Dependencies.py", line 1097, in cythonize
    cythonize_one(*args)
  File "/home/xing/Downloads/software/sage/8.7/local/lib/python2.7/site-packages/Cython/Build/Dependencies.py", line 1220, in cythonize_one
    raise CompileError(None, pyx_file)
Cython.Compiler.Errors.CompileError: gsl_rand.pyx

atexit 正在与 from cython_gsl cimport * 一起导入。这似乎发生在 from libc.stdlib cimport * 行。因此,此名称与 Python atexit 模块冲突。我认为这是一个很好的例子,说明为什么 from something [c]import * 不推荐 ,无论是在您的代码还是在 cython_gsl 的代码中。它变得更加混乱,因为 Cython 有两种类型的名称(Python 和 C),所以你会收到奇怪的错误消息。

最佳解决方案是执行 cimport cython_gsl 或导入您需要的特定符号:from cython_gsl cimport gsl_rng, etc。请记住更改 pyx 和 pxd 文件。

更糟糕的解决方法是在导入时重命名 atexit 模块:import atexit as ae.