Python调用dll

本文介绍了如何在Python中使用ctypes调用C++编译的DLL,包括创建DLL项目、设置参数和返回类型,以及如何传递和接收包含字符数组的结构体。在DLL中,静态字符数组用于避免Python读取乱码,而在Python端,定义相应的结构体类以匹配DLL的结构体定义。

一、在vs2017年创建dll程序

1.新建项目,选择“windows桌面向导”

2.选择应用程序类型为dll,注意还要勾选“空项目”

3.在源文件夹里新建一个cpp,代码如下:

#define DLLEXPORT extern "C" __declspec(dllexport)

#include "stdio.h"



DLLEXPORT wchar_t* sum(wchar_t *str, int b, float f) {

         static wchar_t a[] = L"你好,world";

         static wchar_t szBuffer[100];

         float x = b + f;

         swprintf(szBuffer, 255,L"%s,%s,%f", a, str, x);

         return szBuffer;

}

注意:

1.这里返回的字符数组必须是static的,否则python读取为乱码

 

二.在pycharm里新建python项目并创建一个python文件,代码如下:

from ctypes import *

dll = CDLL('Project8.dll')
dll.sum.argtypes = [c_wchar_p,c_int,c_float]
dll.sum.restype = c_wchar_p
str = "come in";
pchar = dll.sum(str,2,3.215)

print(pchar)

注意:

1.这里主要使用了ctypes包来操作

2.通过CDLL引入dll

3.通过argtypes指定传入dll参数的类型,通过restype指定dll返回数据的类型,否则程序会因不知道数据类型而报错。

4.c_wchar表示宽字符类型,c_wchar_p表示宽字符数组的指针,其他同理。

 

三、向DLL传递结构体或DLL返回结构体

为了让dll和python间能传递结构体,需要在dll定义结构体,在python里定义对应的类

(一)dll的头文件里定义结构体

typedef  struct MyS {

         int a;

         wchar_t *b;

} ms;

(二)dll的源文件代码

#define DLLEXPORT extern "C" __declspec(dllexport)

#include "stdio.h"

#include "MyStruct.h"



DLLEXPORT ms sum(ms m) {

         ms n;

         n.a = m.a * 2;

         static wchar_t wc[100];

         swprintf(wc, 255, L"%s,ok完成2", m.b);

         n.b = wc;

         return n;

}

说明:

1.由于结构体里有字符指针要返回,所以这里先创建了一个static的字符数组,再把地址传给结构体里的字符指针,这样python才能获得该字符串。

(三)python里代码如下:

from ctypes import *



class mss(Structure):

    _fields_=[('a',c_int),

             ('b',c_wchar_p)]





dll = CDLL('Project8.dll')

dll.sum.argtypes = [mss]

dll.sum.restype = mss



x = mss()

x.a = 1

x.b = "你好hello3"

pchar = dll.sum(x)



print(pchar.a)

print(pchar.b)

说明:

1.通过类的方式定义了一个结构体

2.剩余的和上面简单例子一样。

3.这里的x.b是宽字符,因为python默认为utf-8,因此不会出错,但若x.b为窄字符,即c_char_p,注意要转换:x.b = "hello".encode("ansi")

4.在python里定义类时,成员有字符或字符指针的,一定要和dll结构体里的一致,要么全是指针,要么全是字符。

### 如何在Python调用DLL文件 #### 使用 `ctypes` 调用 DLL 文件 `ctypes` 是 Python 中的一个标准库模块,用于加载动态链接库 (DLL) 并调用其中的函数。以下是通过 `ctypes` 加载并调用 DLL 函数的方法。 1. **导入必要的模块** 需要从 `ctypes` 导入相关类来操作 DLL 和数据类型。 2. **指定 DLL 文件路径** 将目标 DLL 的绝对路径传递给 `CDLL` 或 `WinDLL` 类以加载该库。 3. **定义参数和返回值类型** 如果 DLL 函数有特定的数据类型需求,则需设置其输入参数和返回值类型以便正确交互。 4. **调用 DLL 函数** 可像调用普通 Python 函数一样调用已加载的 DLL 函数。 下面是一个完整的示例: ```python from ctypes import * # 定义 DLL 文件路径 dll_path = r"C:\path\to\your\dll.dll" # 加载 DLL 库 lib = CDLL(dll_path) # 假设 DLL 提供了一个名为 add 的函数,接受两个整数作为参数并返回它们的和 lib.add.argtypes = [c_int, c_int] # 设置参数类型为 int, int lib.add.restype = c_int # 设置返回值类型为 int # 调用 DLL 中的 add 函数 result = lib.add(2, 3) print(f"The result of adding 2 and 3 is {result}") # 输出结果应为 5 ``` 此代码片段展示了如何加载一个 DLL 文件并通过它调用一个简单的加法函数[^2]。 #### 处理 C++ 编写的 DLLDLL 是由 C++ 构建时,需要注意符号修饰问题。默认情况下,C++ 编译器会对函数名进行名称修饰(name mangling),这使得 Python 的 `ctypes` 无法直接找到对应的函数。为了避免这一问题,在编写导出函数时需要加上 `extern "C"` 关键字,从而禁用名称修饰[^3]。 例如,假设有一个 C++ 工程中的头文件如下所示: ```cpp #ifdef EXPORTS_DLL #define MY_API __declspec(dllexport) #else #define MY_API __declspec(dllimport) #endif // 确保函数可以被 Python 正确识别 extern "C" MY_API int multiply(int a, int b); ``` 此时可以在 Python 中按照相同方式加载并调用这个乘法函数: ```python from ctypes import * dll_path = r"C:\path\to\your\dll.dll" lib = CDLL(dll_path) lib.multiply.argtypes = [c_int, c_int] lib.multiply.restype = c_int result = lib.multiply(4, 5) print(f"The result of multiplying 4 and 5 is {result}") ``` #### 数据类型的映射 为了确保与 DLL 进行正确的通信,必须清楚地知道每个参数及其对应的结果应该是什么样的数据类型。下表列出了常见的 C 类型到 Python `ctypes` 对象之间的转换关系[^1]: | C Type | ctypes Object | |----------------|--------------------| | char | c_char | | unsigned char | c_ubyte | | short | c_short | | unsigned short | c_ushort | | int | c_int | | unsigned int | c_uint | | long | c_long | | float | c_float | | double | c_double | 对于更复杂的情况,比如字符串处理或者数组传递,可能还需要额外配置缓冲区或指针对象。例如创建字符缓冲区可以用 `create_string_buffer()` 方法实现[^6]。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值