IT狗工作室

第7篇 Cython封装C++代码(中)

2020-05-05  本文已影响0人  铁甲万能狗

至此,我们已经在pyx文件的命名空间中公开了C ++ MoneyFormator类型的接口。 现在,我们需要使它可以从外部Python代码访问(这是我们的重点)在前一篇文章我们通过面向过程的编程方式已经实现了。


常见的编程实践是创建一个Cython扩展类型,该类型将封装后的C ++类实例作为属性,并创建一堆Cython类的方法,这些类方法内部调用C++类实例的对应的方法。 因此,将前一篇中currency.pxd中的Cython类接口导入到一个全新的pyx文件中,这里命名为monfmt.pyx,我们重新创建一个扩展模块,该模块内部就是使用Cython扩展类来调用C++类的接口,如下代码所示。

#distutils:language=c++
#cython:language_level=3

from currency cimport MoneyFormator as _MoneyFormat
from libcpp.string cimport string

#https://stackoverflow.com/questions/13201886/cython-and-constructors-of-classes
cdef class MoneyFormator:

    cdef _MoneyFormat* fmt
    cdef public double digit
        
    def __cinit__(self,str locName="zh_CN.UTF-8",double n=0.0):
        if locName=='zh_CN.UTF-8':
            self.fmt=new _MoneyFormat()
        else:
            self.fmt=new _MoneyFormat(locName.encode('utf-8'))

        self.digit=n
    
    cpdef string format(self,double value):
        self.digit=value
        return self.fmt.str(value)
    
    def __repr__(self):
        if self.fmt!=NULL and self.digit>0:
            return self.fmt.str(self.digit).decode("utf-8")
        else:
            return self.fmt.str(0).decode('utf-8')
    
    def __dealloc__(self):
        if self.fmt!=NULL:
            del self.fmt
         

这个Cython类语法上没什么好说的,我假设你已经阅读过前6篇Cython编程随笔了,在Python代码中调用如下图所示。


ss8.png

Cython类方法的重载问题

唯一注意的是,上面的调用示例,我们刚开始以默认的参数实例化了MoneyFormator对象fmt,我们调用时没有传递参数,但实际上我们调用了一个带参数的构造函数,而非默认构造,其原型是

MoneyFormator. __ cinit __(self,str,double)

其实这里我们要讨论一个额外的问题就是Cython扩展类方法的重载(Overloaded),注意而非重写,读者千万不能将C/C++/Java这类面向对象语言中的重载概念套在Cython扩展类中,因为Cython重载的特性体现两个方面。我们理解Cython重载问题过程中,我们重温一下重载的概念:

重载就是为具有两个或以上具有相同名称但参数不同的函数的过程,在C/C++中称为函数重载。 在函数重载中,通过使用不同类型参数不同数量参数来重新定义函数。 只有通过这些差异,编译器才能区分功能。重载的意义是它可以提高程序的可读性,因为您无需为同一操作使用不同的名称。

首先由关键字def函数定义类方法,它本质上是一个Python函数,意味着所有def关键字定义的类方法,由Python来解析,因为Python的C底层实现结构。决定了无法像原生C/C++那样可以识别不同函数指针

更新中....

上一篇下一篇

猜你喜欢

热点阅读