Cython中的抽象类(使用纯虚方法)

dmy*_*tro 11 c++ python oop cython

快速版: 如何在Cython中声明一个抽象类?目标是仅声明接口,以便其他类可以从中继承,必须没有此类的实现.

interface.pxd:

cdef class IModel:
    cdef void do_smth(self)
Run Code Online (Sandbox Code Playgroud)

impl.pyx:

from interface cimport IModel

cdef class A(IModel):
    cdef void do_smth(self):
        pass
Run Code Online (Sandbox Code Playgroud)

一切都很好编译,但当我impl.so在python中导入时,我得到以下内容:

ImportError: No module named interface
Run Code Online (Sandbox Code Playgroud)

显然这个方法不是真正的虚拟和python想要IModel的实例

更多细节:

我有一个cython扩展类(cdef class Integrator),它应该在任何实例上运行,实现IModel接口.界面只是确保实例具有方法void get_dx(double[:] x, double[:] dx),以便集成商可以在每个集成步骤中调用它,以便集成模型.我们的想法是,可以在cython中实现不同的模型,然后以交互方式集成它们并在python脚本中绘制结果.像那样:

from integrator import Integrator # <-- pre-compiled .so extension
from models import Lorenz         # <-- also pre-compiled one, which inherits
                                  # from IModel

mod = Lorenz()
i = Inegrator(mod)
i.integrate()       # this one's really fast cuz no python is used inside

# do something with data from i
Run Code Online (Sandbox Code Playgroud)

lorenz.pyx级应该是这个样子:

from imodel cimport IModel

cdef class Lorenz(IModel):
    cdef void get_dx(double[:] x, double[:] dx)
        # implementation
Run Code Online (Sandbox Code Playgroud)

而且integrator.pyx:

from imodel cimport IModel

cdef class Integrator:
    cdef IModel model

    def __init__(self, IModel model):
        self.model = model

    # rest of the implementation
Run Code Online (Sandbox Code Playgroud)

理想情况下,IModel应该cython头文件(即imodel.pxd)中以类定义的形式存在,但到目前为止,我只能通过编写一个丑陋的虚拟实现类来实现所需的功能imodel.pyx.最糟糕的是,这个无用的虚拟实现必须被编译和链接,以便其他cython类可以继承它.

PS:我认为这是抽象类的完美用例,但是,如果它对你来说实际上看起来很糟糕,亲爱的OOP编码器,请告诉我我应该使用哪种方法.

dmy*_*tro 6

事实证明,这不太可能(讨论).目前,不支持接口,显然是因为它们不是至关重要的:通常的继承非常好.