c++中的引用和python中的引用_c++ 在Python代码中调用C/C++代码

本文介绍了如何在Python中调用C/C++代码,主要涉及ctypes库和SWIG工具。ctypes适用于直接调用C语言的动态链接库,而SWIG则支持更复杂的接口,并能自动生成包装代码,使得C/C++与Python的交互更加便捷。通过SWIG,可以创建跨平台的Python包,包含C/C++源码,并在setup.py中编译。
摘要由CSDN通过智能技术生成

在Python代码中调用C/C++代码,这需求是比较常见的。 毕竟,当代软件世界的基础设施领域是C语言的天下,很多开发工作不可避免地要与它打交道。 而C++则是家族的嫡长子,也占据了一些不可替代的位置。

从调用的角度细分,还有两种方式: 一是单向调用,只从Python中调用C/C++代码;二是双向调用,C/C++也会回调Python的内容。 此外,从C/C++中调用Python代码,也偶尔出现,因为Python的某些标准库功能非常实用,比如http.server。 大部分情况下,都是从Python单向调用C/C++。

从C/C++代码的编译情况,还能细分成两类: 一是已编译,直接使用C/C++的动态链接库;二是未编译,需要在setup.py中编译成可导入模块。 前者往往用在使用某些著名的库,比如libc.so; 而后者则是往往出现在使用C/C++来提升Python运行效率、或者需要与C/C++交换复杂的数据结构。

本文着重介绍ctypes与SWIG,也会提一下其它方案。

ctypes

对于直接调用C语言的动态链接库,ctypes就是正解。 遗憾的是,它不支持C++。

>>> import ctypes

>>> libc = ctypes.CDLL('libc.so.6')

>>> libc.printf

>>> libc.printf(b'Hello world!\n')

Hello world!

13

>>> libc.time()

1516097322

printf的显示中,Hello world!是打印输出,而13则是printf函数返回值,代表13个打印字符。 详细教程,参考《ctypes tutorial》。 (注意,以上代码直接在python交互环境执行。如果是使用IPython,那么printf的打印结果可能需要退出IPython时才能看到。)

setup.py中打包so

如果不是libc.so这种已安装或易安装的库,而是比较冷门,或者私有的库,就需要和package一起打包。 比如,对于以下结构的一个项目,需要把libxxx.so作为数据打包。

.

├── pkg_name

│ ├── __init__.py

│ └── xxx_wrapper

│ ├── __init__.py

│ └── libxxx.so

└── setup.py

在setup.py中,需要添加以下配置。

setup(

...

package_data={'pkg_name': ['xxx_wrapper/libxxx.so']},

)

SWIGSWIG是个帮助使用C或者C++编写的软件能与其它各种高级编程语言进行嵌入联接的开发工具。 SWIG能应用于各种不同类型的语言包括常用脚本编译语言例如Perl, PHP, Python, Tcl, Ruby and PHP。

写一个interface文件,即可自动生成一个C/C++与一个Python的包装文件。 让连接C/C++与Python的这个抽象层级,完全不用手写。

比如,下面的example.i文件,暴露了四个接口。

%module example

%{

#include "example.h"

%}

extern double My_variable;

extern int fact(int n);

extern int my_mod(int x, int y);

extern char *get_time();

上层的Python可以直接调用这四个接口,而无需考虑中间的转换过程。

import example

print('My_varaiable: %s' % example.cvar.My_variable)

print('fact(5): %s' % example.fact(5))

print('my_mod(7,3): %s' % example.my_mod(7,3))

print('get_time(): %s' % example.get_time())

SWIG与ctypes相比,虽然使用上麻烦一些,但接口更容易使用,也支持C++。 最重要的是,可以很方便地让C/C++代码在setup.py中编译,更容易跨平台。

from setuptools import Extension, setup

EXAMPLE_EXT = Extension(

name='_example',

sources=[

'src/example/example.c',

'src/example/example.i',

],

)

setup(

...

ext_modules=[EXAMPLE_EXT],

)

setuptools中的Extension源于distutils,可以识别*.i文件并自动做出转换。 这样编译的Wheel文件,还自带架构与版本等编译信息,比如:example-0.1.0-cp36-cp36m-linux_x86_64.whl。

同类思想的产物还有SIP,用于PyQt,参考《Using SIP》。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值