Python调用C/C++动态链接库的方法详解


Posted in Python onJuly 22, 2014

本文以实例讲解了Python调用C/C++ DLL动态链接库的方法,具体示例如下:

示例一:

首先,在创建一个DLL工程(本例创建环境为VS 2005),头文件:

//hello.h
#ifdef EXPORT_HELLO_DLL
#define HELLO_API __declspec(dllexport)
#else
#define HELLO_API __declspec(dllimport)
#endif
extern "C"
{
 HELLO_API int IntAdd(int , int);
}

CPP文件:

//hello.cpp
#define EXPORT_HELLO_DLL
#include "hello.h"
HELLO_API int IntAdd(int a, int b)
{
 return a + b;
}

这里有两个注意点:

(1)弄清楚编译的时候函数的调用约定采用的__cdecl还是__stdcall,因为根据DLL中函数调用约定方式,Python将使用相应的函数加载DLL。

(2)如果采用C++的工程,那么导出的接口需要extern "C",这样python中才能识别导出的函数。

我的工程中采用__cdecl函数调用约定方式进行编译链接产生hello.dll,然后Python中采用ctypes库对hello.dll进行加载和函数调用:

from ctypes import *
dll = cdll.LoadLibrary('hello.dll');
ret = dll.IntAdd(2, 4);
print ret;

至此,第一个小例子已经完成了,读者可以自己动手尝试一下运行效果。

示例二:

示例一只是一个"hello world"级别的程序,实际运用中更多的需要传递数据结构、字符串等,才能满足我们的需求。那么本示例将展示,如何传递数据结构参数,以及如何通过数据结构获取返回值。

首先编写DLL工程中的头文件:

//hello.h
#ifdef EXPORT_HELLO_DLL
#define HELLO_API __declspec(dllexport)
#else
#define HELLO_API __declspec(dllimport)
#endif

#define ARRAY_NUMBER 20
#define STR_LEN 20

struct StructTest
{
 int number;
 char* pChar;
 char str[STR_LEN];
 int iArray[ARRAY_NUMBER];
};

extern "C"
{
 //HELLO_API int IntAdd(int , int);
 HELLO_API char* GetStructInfo(struct StructTest* pStruct);
}

CPP文件如下:

//hello.cpp
#include <string.h>
#define EXPORT_HELLO_DLL
#include "hello.h"

HELLO_API char* GetStructInfo(struct StructTest* pStruct)
{
 for (int i = 0; i < ARRAY_NUMBER; i++)
 pStruct->iArray[i] = i;
 pStruct->pChar = "hello python!";
 strcpy (pStruct->str, "hello world!");
 pStruct->number = 100;
 return "just OK";
}

GetStructInfo这个函数通过传递一个StructTest类型的指针,然后对对象中的属性进行赋值,最后返回"just OK".

编写Python调用代码如下,首先在Python中继承Structure构造一个和C DLL中一致的数据结构StructTest,然后设置函数GetStructInfo的参数类型和返回值类型,最后创建一个StructTest对象,并将其转化为指针作为参数,调用函数GetStrcutInfo,最后通过输出数据结构的值来检查是否调用成功

from ctypes import *
ARRAY_NUMBER = 20;
STR_LEN = 20;
#define type
INTARRAY20 = c_int * ARRAY_NUMBER;
CHARARRAY20 = c_char * STR_LEN;
#define struct
class StructTest(Structure):
  _fields_ = [
    ("number", c_int),
    ("pChar", c_char_p),
    ("str", CHARARRAY20),
    ("iArray", INTARRAY20)
        ]
#load dll and get the function object
dll = cdll.LoadLibrary('hello.dll');
GetStructInfo = dll.GetStructInfo;
#set the return type
GetStructInfo.restype = c_char_p;
#set the argtypes
GetStructInfo.argtypes = [POINTER(StructTest)];
objectStruct = StructTest();
#invoke api GetStructInfo
retStr = GetStructInfo(byref(objectStruct));
#check result
print "number: ", objectStruct.number;
print "pChar: ", objectStruct.pChar;
print "str: ", objectStruct.str;
for i,val in enumerate(objectStruct.iArray):
  print 'Array[i]: ', val;
print retStr;

总结:

1. 用64位的Python去加载32位的DLL会出错
2. 以上只是些测试程序,在编写Python过程中尽可能的使用"try Except"来处理异常
3. 注意在Python与C DLL交互的时候字节对齐问题
4. ctypes库的功能还有待继续探索

Python 相关文章推荐
Python实现PS图像调整黑白效果示例
Jan 25 Python
Python中使用Counter进行字典创建以及key数量统计的方法
Jul 06 Python
Python自定义装饰器原理与用法实例分析
Jul 16 Python
利用selenium爬虫抓取数据的基础教程
Jun 10 Python
解决python super()调用多重继承函数的问题
Jun 26 Python
python实现本地批量ping多个IP的方法示例
Aug 07 Python
python3.8下载及安装步骤详解
Jan 15 Python
Python实现不规则图形填充的思路
Feb 02 Python
Python selenium自动化测试模型图解
Apr 15 Python
python实现批量移动文件
Apr 05 Python
python编程学习使用管道Pipe编写优化代码
Nov 20 Python
Python中的turtle画箭头,矩形,五角星
Mar 16 Python
使用python编写批量卸载手机中安装的android应用脚本
Jul 21 #Python
使用python编写脚本获取手机当前应用apk的信息
Jul 21 #Python
使用python编写android截屏脚本双击运行即可
Jul 21 #Python
python 示例分享---逻辑推理编程解决八皇后
Jul 20 #Python
python中from module import * 的一个坑
Jul 20 #Python
用python代码做configure文件
Jul 20 #Python
python中的内置函数getattr()介绍及示例
Jul 20 #Python
You might like
PHP高级对象构建 工厂模式的使用
2012/02/05 PHP
php使用pdo连接mssql server数据库实例
2014/12/25 PHP
thinkphp jquery实现图片上传和预览效果
2020/07/22 PHP
PHP命名空间与自动加载机制的基础介绍
2019/08/25 PHP
jQuery的deferred对象使用详解
2011/08/20 Javascript
jquery键盘事件使用介绍
2011/11/01 Javascript
Javascript判断对象是否相等实现代码
2013/03/18 Javascript
实用框架(iframe)操作代码
2014/10/23 Javascript
JavaScript保存并运算页面中数字类型变量的写法
2015/07/06 Javascript
JS实现超简单的仿QQ折叠菜单效果
2015/09/21 Javascript
深入理解JavaScript单体内置对象
2016/06/06 Javascript
JavaScript判断是否是微信浏览器
2016/06/13 Javascript
vue-router跳转页面的方法
2017/02/09 Javascript
Javascript中字符串相关常用的使用方法总结
2017/03/13 Javascript
JS 实现banner图片轮播效果(鼠标事件)
2017/08/04 Javascript
浅谈vue中慎用style的scoped属性
2017/11/28 Javascript
使用vue-cli编写vue插件的方法
2018/02/26 Javascript
理顺8个版本vue的区别(小结)
2018/09/17 Javascript
vue+egg+jwt实现登录验证的示例代码
2019/05/18 Javascript
Openlayers3实现车辆轨迹回放功能
2020/09/29 Javascript
在Django的模型中执行原始SQL查询的方法
2015/07/21 Python
Linux CentOS Python开发环境搭建教程
2018/11/28 Python
Django中更改默认数据库为mysql的方法示例
2018/12/05 Python
python动态视频下载器的实现方法
2019/09/16 Python
Python restful框架接口开发实现
2020/04/13 Python
在python中使用nohup命令说明
2020/04/16 Python
k-means 聚类算法与Python实现代码
2020/06/01 Python
python代码能做成软件吗
2020/07/24 Python
pycharm 使用tab跳出正在编辑的括号(){}{}等问题
2021/02/26 Python
canvas因为图片资源不在同一域名下而导致的跨域污染画布的解决办法
2019/01/18 HTML / CSS
简述网络文件系统NFS,并说明其作用
2016/10/19 面试题
北京某科技有限公司C# .net笔试题
2014/09/27 面试题
应届生会计求职信
2013/11/11 职场文书
医药学专业大学生职业生涯规划书论文
2014/01/21 职场文书
教师四风对照检查材料思想汇报
2014/09/17 职场文书
MySQL 原理与优化之Limit 查询优化
2022/08/14 MySQL