如何使用 Python 中的 DLL 文件?
- 2025-01-06 08:32:00
- admin 原创
- 122
问题描述:
DLL
使用内部文件的最简单方法是什么Python
?
具体来说,如何在不编写任何额外包装C++
代码来公开功能的情况下做到这一点Python
?
Python
与使用第三方库相比,我们强烈建议使用本机功能。
解决方案 1:
为了易于使用,ctypes是最佳选择。
以下 ctypes 示例来自我编写的实际代码(使用 Python 2.5)。到目前为止,这是我发现的完成您所要求的最简单的方法。
import ctypes
# Load DLL into memory.
hllDll = ctypes.WinDLL ("c:\\PComm\\ehlapi32.dll")
# Set up prototype and parameters for the desired function call.
# HLLAPI
hllApiProto = ctypes.WINFUNCTYPE (
ctypes.c_int, # Return type.
ctypes.c_void_p, # Parameters 1 ...
ctypes.c_void_p,
ctypes.c_void_p,
ctypes.c_void_p) # ... thru 4.
hllApiParams = (1, "p1", 0), (1, "p2", 0), (1, "p3",0), (1, "p4",0),
# Actually map the call ("HLLAPI(...)") to a Python name.
hllApi = hllApiProto (("HLLAPI", hllDll), hllApiParams)
# This is how you can actually call the DLL function.
# Set up the variables and call the Python name with them.
p1 = ctypes.c_int (1)
p2 = ctypes.c_char_p (sessionVar)
p3 = ctypes.c_int (1)
p4 = ctypes.c_int (0)
hllApi (ctypes.byref (p1), p2, ctypes.byref (p3), ctypes.byref (p4))
该ctypes
内容具有所有 C 类型数据类型(int
、、、等等),并且可以通过值或引用传递。它还可以返回特定数据类型,尽管我的示例没有这样做(HLL API 通过修改通过引用传递的变量来返回值)char
。short
`void*`
就上面显示的具体示例而言,IBM 的 EHLLAPI 是一个相当一致的接口。
根据 IBM 的文档,所有调用都传递四个 void 指针(EHLLAPI 通过第四个参数返回返回代码,即指向int
so 的指针,虽然我将其指定为返回类型,但我可以安全地忽略它)。换句话说,该函数的 C 变体将是:int
int hllApi (void *p1, void *p2, void *p3, void *p4)
这使得单个简单的ctypes
函数能够执行 EHLLAPI 库提供的任何功能,但其他库可能需要为ctypes
每个库函数设置单独的函数。
返回值WINFUNCTYPE
是函数原型,但您仍需要设置更多参数信息(超出类型)。 中的每个元组hllApiParams
都有一个参数“方向”(1 = 输入,2 = 输出等)、一个参数名称和一个默认值 -ctypes
有关详细信息,请参阅文档
获得原型和参数信息后,您可以创建一个 Python “可调用”hllApi
来调用该函数。您只需创建所需的变量(在我的情况下为p1
through p4
)并使用它们调用该函数即可。
解决方案 2:
此页面有一个从 DLL 文件调用函数的非常简单的示例。
为了完整性,这里解释一下细节:
在 Python 中调用 DLL 函数非常简单。我有一个自制的 DLL 文件,其中包含两个函数:
add
和,sub
它们接受两个参数。
add(a, b)
返回两个数字的加法
sub(a, b)
返回两个数字的减法DLL 文件的名称将为“demo.dll”
程序:
from ctypes import* # give location of dll mydll = cdll.LoadLibrary("C:\\demo.dll") result1= mydll.add(10,1) result2= mydll.sub(10,1) print("Addition value:"+result1) print("Substraction:"+result2)
输出:
Addition value:11 Substraction:9
解决方案 3:
使用 ctypes 在 Python 下构建 DLL 并链接它
我提供了一个完整的示例,说明如何构建shared library
并在 下使用它Python
。ctypes
我考虑这种Windows
情况并处理DLLs
。需要两个步骤:
从命令行或 IDE 使用 Visual Studio 的编译器构建 DLL;
使用 ctypes 链接 Python 下的 DLL。
共享库
我认为shared library
以下内容包含在testDLL.cpp
文件中。唯一的功能testDLL
只是接收int
并打印它。
#include <stdio.h>
extern "C" {
__declspec(dllexport)
void testDLL(const int i) {
printf("%d
", i);
}
} // extern "C"
从命令行构建 DLL
要DLL
从命令行Visual Studio
运行
"C:Program Files (x86)Microsoft Visual Studio 12.0Common7Toolssdevcmd"
设置包含路径然后运行
cl.exe /D_USRDLL /D_WINDLL testDLL.cpp /MT /link /DLL /OUT:testDLL.dll
构建 DLL。
DLL
从 IDE构建
或者,可以使用如下方法DLL
构建:Visual Studio
文件->新建->项目;
已安装 -> 模板 -> Visual C++ -> Windows -> Win32 -> Win32Project;
下一个;
应用程序类型 -> DLL;
附加选项->空项目(选择);
附加选项->预编译头(取消选择);
项目->属性->配置管理器->活动解决方案平台:x64;
项目->属性->配置管理器->活动解决方案配置:发布。
在 Python 下链接 DLL
在 Python 下,执行以下操作
import os
import sys
from ctypes import *
lib = cdll.LoadLibrary('testDLL.dll')
lib.testDLL(3)
解决方案 4:
或许是Dispatch
:
from win32com.client import Dispatch
zk = Dispatch("zkemkeeper.ZKEM")
其中 zkemkeeper 是系统上注册的 DLL 文件...之后,您只需调用它们即可访问函数:
zk.Connect_Net(IP_address, port)
解决方案 5:
ctypes 是最容易使用的,但(误)使用它会导致 Python 崩溃。如果你想快速做某事,而且你很小心,那么它就很棒。
我鼓励您查看Boost Python。 是的,它要求您编写一些 C++ 代码并拥有 C++ 编译器,但您实际上不需要学习 C++ 即可使用它,而且您可以从 Microsoft 获得免费(如啤酒)C++ 编译器。
解决方案 6:
如果 DLL 是 COM 库类型,那么您可以使用 pythonnet。
pip install pythonnet
然后在你的 Python 代码中尝试以下操作
import clr
clr.AddReference('path_to_your_dll')
# import the namespace and class
from Namespace import Class
# create an object of the class
obj = Class()
# access functions return type using object
value = obj.Function(<arguments>)
然后根据 DLL 中的类实例化一个对象,并访问其中的方法。
扫码咨询,免费领取项目管理大礼包!