python使用ctypes调用扩展模块的实例方法


Posted in Python onJanuary 28, 2020

楔子

我们知道python的执行效率不是很高,而且由于GIL的原因,导致python不能充分利用多核CPU。一般的解决方式是使用多进程,但是多进程开销比较大,而且进程之间的通信也会比较麻烦。因此在解决效率问题上,我们会把那些比较耗时的模块使用C或者C++编写,然后编译成动态链接库,Windows上面是dll,linux上面则是so,编译好之后,交给python去调用。而且通过扩展模块的方式还可以解决python的GIL的问题,因此如果想要利用多核,我们仍然可以通过扩展模块的方式。

python如何调用扩展模块

python调用扩展模块的一种比较简单的方式就是使用ctypes这个库,这个库是python官方提供的,任何一个版本的python都可以使用,我们通过ctypes可以很轻松地调用扩展模块。

演示

#include <stdio.h>

void test()
{
  printf("hello world\n");
}

我们定义了一个很简单的函数,下面我们就可以将其编译成扩展模块了。在Windows是dll,linux上是so,编译的命令是一样的。我这里以Windows 为例,记得在Windows上要安装MinGW,或者安装VsCode,我这里使用的是MinGW,因为VsCode太大了。

gcc -o dll文件或者so文件 -shared c或者c++源文件

我这里的C源文件叫做1.c,我们编译成mmp.dll吧,所以命令就可以这么写:gcc -o mmp.dll -shared 1.c

python使用ctypes调用扩展模块的实例方法

下面就可以使用python去调用了。

import ctypes

# 使用ctypes很简单,直接import进来,然后使用ctypes.CDLL这个类来加载动态模块
# 如果在Windows上还可以使用ctypes.WinDLL。
# 因为看ctypes源码的话,会发现WinDLL也是一个类并且继承自CDLL
# 所以在linux上使用ctypes.CDLL,
# 而在Windows上既可以使用WinDLL、也可以使用CDLL加载动态模块
lib = ctypes.CDLL("./mmp.dll") # 加载之后就得到了扩展模块
# 我们可以直接通过.的方式去调用里面的函数了,会发现成功打印
lib.test() # hello world

# 但是为了确定是否存在这个函数,我们一般会使用反射去获取
# 因为如果函数不存在通过.的方式调用会抛异常的
func = getattr(lib, "test", None)
if func:
  print(func) # <_FuncPtr object at 0x0000029F75F315F0>
  func() # hello world


# 不存在test_xx这个函数,所以得到的结果为None
func1 = getattr(lib, "test_xx", None)
print(func1) # None

所以使用ctypes去调用扩展模块非常方便

1.通过ctypes.CDLL("dll或者so的路径"),如果是Windows还可以使用ctypes.WinDLL("dll路径")。另外这两种加载方式分别等价于:ctypes.CDLL("dll或者so的路径") == ctypes.cdll.LoadLibrary("dll或者so的路径"),ctypes.WinDLL("dll路径") == ctypes.windll.LoadLibrary("dll路径")。但是注意的是:linux上只能使用ctypes.CDLL和ctypes.cdll.LoadLibrary,而Windows上ctypes.CDLL、ctypes.cdll.LoadLibrary、ctypes.WinDLL、ctypes.windll.LoadLibrary都可以使用。但是一般我们都使用ctypes.CDLL即可,另外注意的是:dll或者so文件的路径最好是绝对路径,即便不是也要表明层级,比如我们这里的py文件和dll文件是在同一个目录下,但是我们加载的时候不可以写mmp.dll,这样会报错找不到,要写成./mmp.dll。

2.加载动态模块之后会返回一个对象,我们上面起名为lib,这个lib就是得到的扩展模块了。

3.然后可以直接通过lib调用里面的函数,但是一般我们会使用反射的方式来获取,因为不知道函数到底存不存在,如果不存在直接调用会抛出异常,如果存在这个函数我们才会执行。

以上就是本次介绍的全部相关知识点,如果大家有任何补充的地方可以联系三水点靠木小编。

Python 相关文章推荐
Python 文件操作技巧(File operation) 实例代码分析
Aug 11 Python
python编程实现12306的一个小爬虫实例
Dec 27 Python
Python判断有效的数独算法示例
Feb 23 Python
Python数据可视化:泊松分布详解
Dec 07 Python
Spring Cloud Feign高级应用实例详解
Dec 10 Python
python 实现让字典的value 成为列表
Dec 16 Python
python使用HTMLTestRunner导出饼图分析报告的方法
Dec 30 Python
Python3.6安装卸载、执行命令、执行py文件的方法详解
Feb 20 Python
matplotlib 曲线图 和 折线图 plt.plot()实例
Apr 17 Python
解决Alexnet训练模型在每个epoch中准确率和loss都会一升一降问题
Jun 17 Python
python 将列表里的字典元素合并为一个字典实例
Sep 01 Python
Python语言编写智力问答小游戏功能
Oct 13 Python
Python 时间戳之获取整点凌晨时间戳的操作方法
Jan 28 #Python
使用Python制作新型冠状病毒实时疫情图
Jan 28 #Python
代码总结Python2 和 Python3 字符串的区别
Jan 28 #Python
使用Python爬虫库BeautifulSoup遍历文档树并对标签进行操作详解
Jan 25 #Python
Python爬虫库BeautifulSoup获取对象(标签)名,属性,内容,注释
Jan 25 #Python
Python爬虫库BeautifulSoup的介绍与简单使用实例
Jan 25 #Python
使用Python爬虫库requests发送表单数据和JSON数据
Jan 25 #Python
You might like
php date()日期时间函数详解
2010/05/16 PHP
php实现将上传word文件转为html的方法
2015/06/03 PHP
php实现基于pdo的事务处理方法示例
2017/07/21 PHP
ThinkPHP5+Layui实现图片上传加预览功能
2018/08/17 PHP
浅谈laravel orm 中的一对多关系 hasMany
2019/10/21 PHP
laravel框架模型中非静态方法也能静态调用的原理分析
2019/11/23 PHP
索趣科技的答案
2007/02/07 Javascript
从sohu弄下来的flash中展示图片的代码
2007/04/27 Javascript
瀑布流布局代码一例
2014/04/11 Javascript
node.js中的console.trace方法使用说明
2014/12/09 Javascript
JavaScript事件委托用法分析
2015/01/24 Javascript
基于Css3和JQuery实现打字机效果
2015/08/11 Javascript
JS实现漂亮的时间选择框效果
2016/08/20 Javascript
js+css3制作时钟特效
2016/10/16 Javascript
详解NodeJs支付宝移动支付签名及验签
2017/01/06 NodeJs
Javascript面试经典套路reduce函数查重
2017/03/23 Javascript
Vue AST源码解析第一篇
2017/07/19 Javascript
基于JavaScript实现无缝滚动效果
2017/07/21 Javascript
vue表单绑定实现多选框和下拉列表的实例
2017/08/12 Javascript
JS设计模式之观察者模式实现实时改变页面中金额数的方法
2018/02/05 Javascript
vue+axios 前端实现的常用拦截的代码示例
2018/08/23 Javascript
Python找出list中最常出现元素的方法
2016/06/14 Python
shell命令行,一键创建 python 模板文件脚本方法
2018/03/20 Python
Pyqt5如何让QMessageBox按钮显示中文示例代码
2019/04/11 Python
python join方法使用详解
2019/07/30 Python
pycharm安装及如何导入numpy
2020/04/03 Python
Python-jenkins模块之folder相关操作介绍
2020/05/12 Python
利用Python实现自动扫雷小脚本
2020/12/17 Python
简述Linux文件系统通过i节点把文件的逻辑结构和物理结构转换的工作过程
2016/01/06 面试题
教师敬业奉献模范事迹材料
2014/05/18 职场文书
学校学雷锋活动总结
2014/06/26 职场文书
优秀党支部申报材料
2014/12/24 职场文书
小王子读书笔记
2015/06/29 职场文书
话题作文之关于呼唤
2019/11/29 职场文书
十个Python自动化常用操作,即拿即用
2021/05/10 Python
Node.js实现断点续传
2021/06/23 Javascript