关于C和Python之间通信的方法

时间:2022-01-06 11:12:16

这些天在做一个项目,前端使用python来作,便于快速开发,底层使用C来写,效率高,速度快。这样就牵扯到二者之间的通信问题。我查了相关文档,发现有以下几种解决方式:

  1. 使用标准的数据表示形式,比如xml或者xdr,然后在C和python两端都有自己的相应的库,可以对这些数据进行解释,自然就可以通信了。
  2. 在C 中定义一个结构对象,把它打包成二进制形式(python把它作为字符串来解释),然后使用python中的struct模块的pack函数来解析,这样 就牵扯到一个解析格式串的问题,一个简单的方式是在c和python两端定义对应的两套数据结构,分别有自己的pack和unpack函数,它们可以对中 间结果---二进制串,进行解析,这样就可以实现通信了。在python那一端还可以使用array模块,处理类型统一的数据,特别方便,有时候比用 struct模块要爽!
  3. 使用一个xml文件将C中定义的结构都包含进去,然后在python那一端进行解析这个xml文件,自然就知道如何对每个C中的struct对象进行解析了,这样扩展性好一些。
  4. 使用第三方的库,我所知道的有boost.python和ctypes,具体怎样我也没有是用。

我在实现的时候使用了第2种方式,下面举个例子:

在c中有这样一个结构:

typedef struct test_tag {         int a;         int b; }test_t;

char* test_pack(test_t* ptr){          char * p=NULL;          p = (char *) malloc(sizeof(test_t));          memcpy(p,ptr,sizeof(test_t));          return p; }

test_t* test_unpack(char* ptr){          test_t* p=NULL;          p=(test_t *)malloc(sizeof(test_t));          memcpy(p,ptr,sizeof(test_t));          return p; }

在python端有这样的对应数据结构:

class test:            format='2i'            members=('a','b')            def __init__(self):                   for item in test.members:                           self.__dict__[item]=-1                          def pack(self,order='@'):                         return struct.pack(order+test.format,self.a,self.b)

             def pack2(self,order='@'):                         bin=array.array('l')                         for item in test.members:                                bin.append(self.__dict__[item])                         if (sys.byteorder=='little' and order=='>') or (sys.byteorder=='big' and order=='<'):                                  bin.byteswap()                         return bin.tostring()                              def unpack(self,data,order='@'):                         (self.a,self.b)=struct.unpack(order+test.format,data)

              def unpack2(self,data,order='@'):                         bin=array.array('l')                         bin.fromstring(data)                         for i,item in enumerate(test.members):                               self.__dict__[item]=bin[i]

然后就可以使用这些函数进行通信了