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基础教程之数字处理(math)模块详解
Mar 25 Python
使用python实现正则匹配检索远端FTP目录下的文件
Mar 25 Python
Python选择排序、冒泡排序、合并排序代码实例
Apr 10 Python
Python实现把回车符\r\n转换成\n
Apr 23 Python
Python  unittest单元测试框架的使用
Sep 08 Python
python pygame实现方向键控制小球
May 17 Python
用python求一个数组的和与平均值的实现方法
Jun 29 Python
12个步骤教你理解Python装饰器
Jul 01 Python
命令行运行Python脚本时传入参数的三种方式详解
Oct 11 Python
Python如何访问字符串中的值
Feb 09 Python
Pandas DataFrame求差集的示例代码
Dec 13 Python
Python爬虫:从m3u8文件里提取小视频的正确操作
May 14 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
《超神学院》霸气归来, 天使彦上演维多利亚的秘密
2020/03/02 国漫
PHP 查找字符串常用函数介绍
2012/06/07 PHP
YII2框架使用控制台命令的方法分析
2020/03/18 PHP
静态页面的值传递(三部曲)
2006/09/25 Javascript
functional继承模式 摘自javascript:the good parts
2011/06/20 Javascript
Js+Flash实现访问剪切板操作
2012/11/20 Javascript
Jquery 复选框取值兼容FF和IE8(测试有效)
2013/10/29 Javascript
纯javascript实现四方向文本无缝滚动效果
2015/06/16 Javascript
js实现选中页面文字将其分享到新浪微博
2015/11/05 Javascript
JavaScript中eval()函数用法详解
2015/12/14 Javascript
基于JavaScript实现根据手机定位获取当前具体位置(X省X市X县X街道X号)
2015/12/29 Javascript
bootstrap table动态加载数据示例代码
2017/03/25 Javascript
jQueryeasyui 中如何使用datetimebox 取两个日期间相隔的天数
2017/06/13 jQuery
Angularjs的键盘事件的绑定
2017/07/27 Javascript
[02:25]专访DOTA2负责人Erik 国际邀请赛暂不会离开西雅
2014/07/21 DOTA
Python聚类算法之凝聚层次聚类实例分析
2015/11/20 Python
python 简单搭建阻塞式单进程,多进程,多线程服务的实例
2017/11/01 Python
Python RabbitMQ消息队列实现rpc
2018/05/30 Python
浅谈python的深浅拷贝以及fromkeys的用法
2019/03/08 Python
python画图——实现在图上标注上具体数值的方法
2019/07/08 Python
python3实现mysql导出excel的方法
2019/07/31 Python
Python Opencv实现单目标检测的示例代码
2020/09/08 Python
python中Mako库实例用法
2020/12/31 Python
浅析Python打包时包含静态文件处理方法
2021/01/15 Python
迪拜领先运动补剂零售品牌中文站:Sporter商城
2019/08/20 全球购物
美国Max仓库:Max Warehouse
2020/05/31 全球购物
3个CCIE对一个工程师的面试题
2012/05/06 面试题
农民工创业典型事迹
2014/01/25 职场文书
十佳好少年事迹材料
2014/08/21 职场文书
工作态度不端正检讨书
2014/10/04 职场文书
行政文员岗位职责
2015/02/04 职场文书
交通事故代理词范文
2015/05/23 职场文书
2015暑假打工实践报告
2015/07/13 职场文书
Django中session进行权限管理的使用
2021/07/09 Python
Javascript中Microtask和Macrotask鲜为人知的知识点
2022/04/02 Javascript
Windows Server 2008配置防火墙策略详解
2022/06/28 Servers