Python调用C/C++动态链接库的方法详解
本文以实例讲解了Python调用C/C++ DLL动态链接库的方法,具体示例如下:
示例一:
首先,在创建一个DLL工程(本例创建环境为VS 2005),头文件:
| 1 2 3 4 5 6 7 8 9 10 | //hello.h#ifdef EXPORT_HELLO_DLL#define HELLO_API __declspec(dllexport)#else#define HELLO_API __declspec(dllimport)#endifextern"C"{ HELLO_API intIntAdd(int, int);} | 
CPP文件:
| 1 2 3 4 5 6 7 | //hello.cpp#define EXPORT_HELLO_DLL#include "hello.h"HELLO_API intIntAdd(inta, intb){ returna + b;} | 
这里有两个注意点:
(1)弄清楚编译的时候函数的调用约定采用的__cdecl还是__stdcall,因为根据DLL中函数调用约定方式,Python将使用相应的函数加载DLL。
(2)如果采用C++的工程,那么导出的接口需要extern "C",这样python中才能识别导出的函数。
我的工程中采用__cdecl函数调用约定方式进行编译链接产生hello.dll,然后Python中采用ctypes库对hello.dll进行加载和函数调用:
| 1 2 3 4 | from ctypes import *dll = cdll.LoadLibrary('hello.dll');ret = dll.IntAdd(2, 4);print ret; | 
至此,第一个小例子已经完成了,读者可以自己动手尝试一下运行效果。
示例二:
示例一只是一个"hello world"级别的程序,实际运用中更多的需要传递数据结构、字符串等,才能满足我们的需求。那么本示例将展示,如何传递数据结构参数,以及如何通过数据结构获取返回值。
首先编写DLL工程中的头文件:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 | //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 20structStructTest{ intnumber; char* pChar; charstr[STR_LEN]; intiArray[ARRAY_NUMBER];};extern"C"{ //HELLO_API int IntAdd(int , int); HELLO_API char* GetStructInfo(structStructTest* pStruct);} | 
CPP文件如下:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 | //hello.cpp#include <string.h>#define EXPORT_HELLO_DLL#include "hello.h"HELLO_API char* GetStructInfo(structStructTest* pStruct){ for(inti = 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,最后通过输出数据结构的值来检查是否调用成功:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 | fromctypes import*ARRAY_NUMBER =20;STR_LEN =20;#define typeINTARRAY20 =c_int *ARRAY_NUMBER;CHARARRAY20 =c_char *STR_LEN;#define structclassStructTest(Structure):  _fields_ =[    ("number", c_int),    ("pChar", c_char_p),    ("str", CHARARRAY20),    ("iArray", INTARRAY20)        ]#load dll and get the function objectdll =cdll.LoadLibrary('hello.dll');GetStructInfo =dll.GetStructInfo;#set the return typeGetStructInfo.restype =c_char_p;#set the argtypesGetStructInfo.argtypes =[POINTER(StructTest)];objectStruct =StructTest();#invoke api GetStructInforetStr =GetStructInfo(byref(objectStruct));#check resultprint"number: ", objectStruct.number;print"pChar: ", objectStruct.pChar;print"str: ", objectStruct.str;fori,val inenumerate(objectStruct.iArray):  print'Array[i]: ', val;printretStr; | 
总结:
1. 用64位的Python去加载32位的DLL会出错
2. 以上只是些测试程序,在编写Python过程中尽可能的使用"try Except"来处理异常
3. 注意在Python与C DLL交互的时候字节对齐问题
4. ctypes库的功能还有待继续探索
 
                     
                    
                 
                    
                
 
 
                
            
         
         浙公网安备 33010602011771号
浙公网安备 33010602011771号