与C/C++库交互

https://farm5.staticflickr.com/4173/34725951345_c8f5959a2e_k_d.jpg

C语言外部函数接口(CFFI)

CFFI 通过CPython和PyPy给出了和 C语言交互的简单使用机制。它支持两种模式:一种是内联的ABI兼容模式(示例如下), 它允许您动态加载和运行可执行模块的函数(本质上与LoadLibrary和dlopen拥有相同的功能); 另一种为API模式,它允许您构建C语言扩展模块。

ABI 交互

1
2
3
4
5
6
7
from cffi import FFI
ffi = FFI()
ffi.cdef("size_t strlen(const char*);")
clib = ffi.dlopen(None)
length = clib.strlen("String to be evaluated.")
# prints: 23
print("{}".format(length))

ctypes

ctypes 是CPython中与C/C++ 交互的事实上的库。它不仅能完全访问大多数主流操作系统(比如:Windows上的Kernel32, *nix上的libc)的纯C接口,并且支持对动态库的加载和交互,如DLL和运行时共享对象。 它同时涵盖许多可和系统API交互的类型,并允许您以相对简单的方式定义自己的复杂类型, 如struct和union,并在需要时允许您作出如填充、对齐这样的修改。对它的使用可能稍显复杂, 但与 struct 模块配合使用, 可通过纯C(++)方法让您从根本上控制您的数据类型转换成更有用的东西。

Struct Equivalents

MyStruct.h

1
2
3
4
struct my_struct {
    int a;
    int b;
};

MyStruct.py

1
2
3
4
import ctypes
class my_struct(ctypes.Structure):
    _fields_ = [("a", c_int),
                ("b", c_int)]

SWIG

SWIG 并不仅仅应用于Python(它支持多种脚本语言), 它是生成解释性语言和C/C++头文件绑定的工具。它极易使用:使用者只需简单的定义接口文件 (详见相关指南和文档),包含必要的C/C++头文件,并对它们运行生成工具。但它也有其局限性, 目前,它与C++部分新特性间仍存在问题,而模板重码的工作多少有些冗繁。只需少量的工作, 它便能提供诸多作用,并展现Python的许多特性。同时,您可以简单的扩展SWIG生成的绑定 (在接口文件中)来重载操作符和内建函数,也可以有效的重新转换C++异常, 使其可被Python所捕获。

例子: Overloading __repr__

MyClass.h

1
2
3
4
5
6
7
#include <string>
class MyClass {
private:
    std::string name;
public:
    std::string getName();
};

myclass.i

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
%include "string.i"

%module myclass
%{
#include <string>
#include "MyClass.h"
%}

%extend MyClass {
    std::string __repr__()
    {
        return $self->getName();
    }
}

%include "MyClass.h"

Boost.Python

Boost.Python 需要一些手动工作来展现C++对象的功能,但它可提供SWIG拥有的所有特性。同时, 它可提供在C++中访问Python对象的封装,也可提取SWIG封装的对象, 甚至可在C++代码中嵌入部分Python。