至此,我们已经在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代码中调用如下图所示。
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++那样可以识别不同函数指针
更新中....