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的else子句使用指南
Feb 27 Python
浅谈python抛出异常、自定义异常, 传递异常
Jun 20 Python
python中日志logging模块的性能及多进程详解
Jul 18 Python
Python实现重建二叉树的三种方法详解
Jun 23 Python
Django框架实现的普通登录案例【使用POST方法】
May 15 Python
使用Python-OpenCV向图片添加噪声的实现(高斯噪声、椒盐噪声)
May 28 Python
Python 实现数据结构-堆栈和队列的操作方法
Jul 17 Python
python3.7环境下安装Anaconda的教程图解
Sep 10 Python
浅谈Django2.0 加xadmin踩的坑
Nov 15 Python
TFRecord文件查看包含的所有Features代码
Feb 17 Python
使用anaconda安装pytorch的实现步骤
Sep 03 Python
PYTHON基于Pyecharts绘制常见的直角坐标系图表
Apr 28 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
星际争霸 Starcraft 游戏介绍
2020/03/14 星际争霸
咖啡历史、消费和行业趋势
2021/03/03 咖啡文化
php学习之运算符相关概念
2011/06/09 PHP
php实现的双向队列类实例
2014/09/24 PHP
Yii框架弹出窗口组件CJuiDialog用法分析
2017/01/07 PHP
PHP结合Ffmpeg快速搭建流媒体服务的实践记录
2018/10/31 PHP
js自动闭合html标签(自动补全html标记)
2012/10/04 Javascript
jquery判断复选框是否被选中的方法
2015/10/16 Javascript
JS动态增删表格行的方法
2016/03/03 Javascript
Bootstrap3 datetimepicker控件使用实例
2016/12/13 Javascript
jQuery+vue.js实现的九宫格拼图游戏完整实例【附源码下载】
2017/09/12 jQuery
Vue+mui实现图片的本地缓存示例代码
2018/05/24 Javascript
vue、react等单页面项目部署到服务器的方法及vue和react的区别
2018/09/29 Javascript
Angular7创建项目、组件、服务以及服务的使用
2019/02/19 Javascript
JS实现简单的表格增删
2020/01/16 Javascript
JavaScript检测是否开启了控制台(F12调试工具)
2020/10/02 Javascript
[06:42]DOTA2每周TOP10 精彩击杀集锦vol.1
2014/06/25 DOTA
[52:08]DOTA2上海特级锦标赛主赛事日 - 3 败者组第三轮#2Fnatic VS OG第一局
2016/03/05 DOTA
[31:47]夜魇凡尔赛茶话会 第三期01:选手知多少
2021/03/11 DOTA
[01:10:24]DOTA2-DPC中国联赛 正赛 VG vs Aster BO3 第一场 2月28日
2021/03/11 DOTA
python爬虫入门教程--快速理解HTTP协议(一)
2017/05/25 Python
快速了解Python开发中的cookie及简单代码示例
2018/01/17 Python
ubuntu安装sublime3并配置python3环境的方法
2018/03/15 Python
Python基于win32ui模块创建弹出式菜单示例
2018/05/09 Python
wxPython之wx.DC绘制形状
2019/11/19 Python
python实现简单颜色识别程序
2020/02/19 Python
TensorFlow tf.nn.softmax_cross_entropy_with_logits的用法
2020/04/19 Python
Django REST Swagger实现指定api参数
2020/07/07 Python
Python如何将字符串转换为日期
2020/07/31 Python
利用python实现后端写网页(flask框架)
2021/02/28 Python
详解HTML5中的元素与元素
2015/08/17 HTML / CSS
VSCode 自定义html5模板的实现
2019/12/05 HTML / CSS
说说你所熟悉或听说过的j2ee中的几种常用模式?及对设计模式的一些看法
2012/05/24 面试题
四议两公开实施方案
2014/03/28 职场文书
第二批党的群众路线教育实践活动个人对照检查材料
2014/09/23 职场文书
浅谈redis整数集为什么不能降级
2021/07/25 Redis