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模块学习 datetime介绍
Aug 27 Python
Python中的ceil()方法使用教程
May 14 Python
python实用代码片段收集贴
Jun 03 Python
django数据库migrate失败的解决方法解析
Feb 08 Python
Python实现输出某区间范围内全部素数的方法
May 02 Python
Python机器学习之scikit-learn库中KNN算法的封装与使用方法
Dec 14 Python
Python中栈、队列与优先级队列的实现方法
Jun 30 Python
Python DataFrame一列拆成多列以及一行拆成多行
Aug 06 Python
使用pytorch实现可视化中间层的结果
Dec 30 Python
Python魔法方法 容器部方法详解
Jan 02 Python
使用Tensorboard工具查看Loss损失率
Feb 15 Python
python wsgiref源码解析
Feb 06 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 daddslashes 使用方法介绍
2012/10/26 PHP
php实现QQ空间获取当前用户的用户名并生成图片
2015/07/25 PHP
对比分析php中Cookie与Session的异同
2016/02/19 PHP
非集成环境的php运行环境(Apache配置、Mysql)搭建安装图文教程
2016/04/12 PHP
深入理解PHP中的empty和isset函数
2016/05/26 PHP
ZendFramework框架实现连接两个或多个数据库的方法
2016/12/08 PHP
利用NodeJS和PhantomJS抓取网站页面信息以及网站截图
2013/11/18 NodeJs
Extjs grid添加一个图片状态或者按钮的方法
2014/04/03 Javascript
Node.js 制作实时多人游戏框架
2015/01/08 Javascript
纯js代码制作的网页时钟特效【附实例】
2016/03/30 Javascript
微信小程序上滑加载下拉刷新(onscrollLower)分批加载数据(二)
2017/05/11 Javascript
解决vue-cli单页面手机应用input点击手机端虚拟键盘弹出盖住input问题
2018/08/25 Javascript
nuxt.js中间件实现拦截权限判断的方法
2018/11/21 Javascript
浅谈vue单页面中有多个echarts图表时的公用代码写法
2020/07/19 Javascript
vuex 多模块时 模块内部的mutation和action的调用方式
2020/07/24 Javascript
jquery实现加载更多&quot;转圈圈&quot;效果(示例代码)
2020/11/09 jQuery
Python使用scrapy抓取网站sitemap信息的方法
2015/04/08 Python
python使用装饰器和线程限制函数执行时间的方法
2015/04/18 Python
在Python的Django框架中显示对象子集的方法
2015/07/21 Python
Python实现将SQLite中的数据直接输出为CVS的方法示例
2017/07/13 Python
python实现报表自动化详解
2017/11/16 Python
python flask web服务实现更换默认端口和IP的方法
2019/07/26 Python
Django REST framework内置路由用法
2019/07/26 Python
python实现word文档批量转成自定义格式的excel文档的思路及实例代码
2020/02/21 Python
Django中的模型类设计及展示示例详解
2020/05/29 Python
keras在构建LSTM模型时对变长序列的处理操作
2020/06/29 Python
美国女鞋品牌:naturalizer(娜然)
2016/08/01 全球购物
英国女鞋购物网站:Moda in Pelle
2019/02/18 全球购物
测量实习生自我鉴定
2013/09/19 职场文书
产品质量承诺书
2014/03/27 职场文书
2015秋学期开学寄语
2015/05/28 职场文书
太行山上观后感
2015/06/05 职场文书
党纪处分决定书
2015/06/24 职场文书
2016新年感言
2015/08/03 职场文书
mysql中整数数据类型tinyint详解
2021/12/06 MySQL
Python获取字典中某个key的value
2022/04/13 Python