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 生成目录树及显示文件大小的代码
Jul 23 Python
python语言使用技巧分享
May 31 Python
Python正则表达式常用函数总结
Jun 24 Python
PyQt5实现简易计算器
May 30 Python
在pycharm下设置自己的个性模版方法
Jul 15 Python
Django-Model数据库操作(增删改查、连表结构)详解
Jul 17 Python
python多线程实现TCP服务端
Sep 03 Python
python 五子棋如何获得鼠标点击坐标
Nov 04 Python
Python上下文管理器全实例详解
Nov 12 Python
python range实例用法分享
Feb 06 Python
python mysql 字段与关键字冲突的解决方式
Mar 02 Python
OpenCV-Python实现油画效果的实例
Jun 08 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文本数据库的搜索方法
2006/10/09 PHP
使用无限生命期Session的方法
2006/10/09 PHP
Laravel中使用阿里云OSS Composer包分享
2015/02/10 PHP
简单谈谈PHP中的trait
2017/02/25 PHP
Thinkphp5 微信公众号token验证不成功的原因及解决方法
2017/11/12 PHP
基于jquery的tab切换 js原理
2010/04/01 Javascript
初试jQuery EasyUI 使用介绍
2010/04/01 Javascript
php与js的区别是什么
2013/08/05 Javascript
jquery做的一个简单的屏幕锁定提示框
2014/03/26 Javascript
js实现选中复选框文字变色的方法
2015/08/14 Javascript
jQuery实现的仿select功能代码
2015/08/19 Javascript
深入分析jsonp协议原理
2015/09/26 Javascript
JS截取字符串实例详解
2015/11/24 Javascript
JS基于ocanvas插件实现的简单画板效果代码(附demo源码下载)
2016/04/05 Javascript
基于jQuery实现音乐播放试听列表
2016/04/14 Javascript
jQuery ajax调用后台aspx后台文件的两种常见方法(不是ashx)
2016/06/28 Javascript
jquery UI Datepicker时间控件冲突问题解决
2016/12/16 Javascript
通过npm引用的vue组件使用详解
2017/03/02 Javascript
JS简单验证上传文件类型的方法
2017/04/17 Javascript
解决Vue-Router升级导致的Uncaught (in promise)问题
2020/08/07 Javascript
jQuery实现增删改查
2020/12/22 jQuery
[01:11]steam端dota2实名认证操作流程视频
2021/03/11 DOTA
对python同一个文件夹里面不同.py文件的交叉引用方法详解
2018/12/15 Python
python几种常用功能实现代码实例
2019/12/25 Python
Python的Tqdm模块实现进度条配置
2021/02/24 Python
Sisley法国希思黎美国官方网站:享誉全球的奢华植物美容品牌
2020/06/27 全球购物
办公室文员工作自我评价
2013/12/01 职场文书
医院护士见习期自我鉴定
2014/04/10 职场文书
数控技校生自我鉴定
2014/04/19 职场文书
艺术设计专业求职自荐信
2014/05/19 职场文书
化学教育专业求职信
2014/07/08 职场文书
运动会广播稿200字(10篇)
2014/10/12 职场文书
张家口市高新区党工委群众路线教育实践活动整改方案
2014/10/25 职场文书
教师旷工检讨书
2015/08/15 职场文书
评测 | 大屏显示带收音机的高端音箱,JBL TUNE2便携式插卡音箱实测
2021/04/24 无线电
redis通过6379端口无法连接服务器(redis-server.exe闪退)
2021/05/08 Redis