5

我有一些目前看起来的 Cython 代码

exc = sys.exc_info()
raise exc[0], exc[1], exc[2]

这在 Python3 上不起作用,因为不再允许“从元组引发”形式。如果这是普通的 Python 代码,我只会使用six.reraise,但我在这里不可用。什么是 Cython 友好的方式来做同样的事情,它适用于 Python2 和 Python3?

4

1 回答 1

4

Cython 的一个很棒的功能是生成的 C 代码可以针对 Python 2 或 Python 3 进行编译。因此,您上面的示例将适用于任何版本的 Python,无需修改。

您可以告诉 Cython 编译代码假设 Python 2 语法和语义(-2参数,默认情况下启用)或假设 Python 3(-3参数)。无论哪种情况,只要动态组件(导入等)兼容,生成的扩展模块源代码都可以编译并用于 Python 2 或 Python 3。

例如:

def raises_exception():
    raise KeyError("what you doin'?")


def foobar():
    try:
        raises_exception()
    except Exception:
        import sys
        exc = sys.exc_info()
        raise exc[0], exc[1], exc[2]

这是一个setup.py适用于 Py2 或 Py3 的:

from distutils.core import setup
from Cython.Build import cythonize

setup(ext_modules=cythonize("reraise.pyx"))

我可以python setup.py build_ext -i在任一版本的 Python 上运行(前提是我为每个版本都安装了 cython),并且生成的扩展模块可以工作。

$ python setup.py build_ext -i # Py3 python

$ ipython3
Python 3.3.2 (v3.3.2:d047928ae3f6, Oct  4 2013, 15:49:17)
Type "copyright", "credits" or "license" for more information.

IPython 1.2.1 -- An enhanced Interactive Python.
?         -> Introduction and overview of IPython's features.
%quickref -> Quick reference.
help      -> Python's own help system.
object?   -> Details about 'object', use 'object??' for extra details.

In [1]: import reraise

In [2]: reraise.foobar()
---------------------------------------------------------------------------
KeyError                                  Traceback (most recent call last)
<ipython-input-2-9e20eacfd84e> in <module>()
----> 1 reraise.foobar()

/.../reraise.so in reraise.foobar (reraise.c:916)()

/.../reraise.so in reraise.foobar (reraise.c:847)()

/.../reraise.so in reraise.raises_exception (reraise.c:762)()

KeyError: "what you doin'?"

In [3]:
于 2014-05-15T15:26:39.847 回答