如何使用 Python 中的 DLL 文件?

Posted

技术标签:

【中文标题】如何使用 Python 中的 DLL 文件?【英文标题】:How can I use a DLL file from Python? 【发布时间】:2010-09-20 03:02:15 【问题描述】:

Python 中使用DLL 文件的最简单方法是什么?

具体来说,如何做到这一点编写任何额外的包装器C++ 代码以将功能公开给Python

本机 Python 功能比使用第三方库更受青睐。

【问题讨论】:

@Remuze 您是否标记了错误的问题或标记了错误的答案?它们都是 python,但我看不出为什么你会发现它们完全相似(我检查了修订历史以确保它在过去的某个时候没有更接近) @foon hmmm 是的,对不起,我的意思是把另一个问题标记为重复,然后放这个 :) 我想这表明它们有多相似! 【参考方案1】:

如果DLL是COM库类型,那么你可以使用pythonnet。

pip install pythonnet

然后在您的 python 代码中,尝试以下操作

import clr
clr.AddReference('path_to_your_dll')

然后根据 DLL 中的类实例化一个对象,并访问其中的方法。

【讨论】:

请注意:当前版本的 pythonnet 不适用于 .NET Core (github.com/pythonnet/pythonnet/issues/984) @gdyrrahitis,就我而言,我使用过 .NET 框架,感谢您提供的信息。【参考方案2】:

在 Python 下使用 ctypes 构建 DLL 并链接它

我提供了一个完整的示例,说明如何通过ctypes 构建shared library 并在Python 下使用它。我考虑Windows 的情况并处理DLLs。需要两个步骤:

    使用 Visual Studio 的编译器从命令行或 IDE 构建 DLL; 使用 ctypes 在 Python 下链接 DLL。

共享库

我考虑的shared library 如下,包含在testDLL.cpp 文件中。唯一的函数testDLL 只是接收int 并打印它。

#include <stdio.h>
​
extern "C" 
​
__declspec(dllexport)
​
void testDLL(const int i) 
    printf("%d\n", i);

​
 // extern "C"

从命令行构建 DLL

从命令行运行使用Visual Studio 构建DLL

"C:\Program Files (x86)\Microsoft Visual Studio 12.0\Common7\Tools\vsdevcmd"

设置包含路径然后运行

cl.exe /D_USRDLL /D_WINDLL testDLL.cpp /MT /link /DLL /OUT:testDLL.dll

构建 DLL。

从 IDE 构建 DLL

或者,可以使用Visual Studio 构建DLL,如下所示:

    文件 -> 新建 -> 项目; 已安装 -> 模板 -> 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)

【讨论】:

如果python包中的DLL怎么办?我如何访问它?【参考方案3】:

This page 有一个从 DLL 文件调用函数的非常简单的示例。

为了完整起见,在此处解释详细信息:

在 Python 中调用 DLL 函数非常容易。我有一个自制的 DLL 文件,它有两个函数:addsub,它们有两个参数。

add(a, b) 返回两个数字相加sub(a, b) 返回两个数字相减

DLL 文件的名称将是“demo.dll”

计划:

from ctypes import*# give location of dllmydll = cdll.LoadLibrary("C:\\demo.dll")result1= mydll.add(10,1)result2= mydll.sub(10,1)print "Addition value:"+result1print "Substraction:"+result2

输出:

Addition value:11Substraction:9

【讨论】:

此答案中的链接已损坏。 如果链接不起作用,请不要担心,这是直接复制粘贴。您没有遗漏任何信息。【参考方案4】:

为了便于使用,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 类型的数据类型(intcharshortvoid* 等),并且可以按值或引用传递。它也可以返回特定的数据类型,尽管我的示例没有这样做(HLL API 通过修改通过引用传递的变量来返回值)。


就上图的具体示例而言,IBM 的 EHLLAPI 是一个相当一致的接口。

所有调用都传递四个 void 指针(EHLLAPI 通过第四个参数发回返回码,一个指向 int 的指针,因此,当我指定 int 作为返回类型时,我可以放心地忽略它)按照 IBM 的文档here。换句话说,该函数的 C 变体将是:

int hllApi (void *p1, void *p2, void *p3, void *p4)

这使得单个简单的 ctypes 函数能够执行 EHLLAPI 库提供的任何操作,但其他库可能需要为每个库函数设置单独的 ctypes 函数。

WINFUNCTYPE 的返回值是一个函数原型,但您仍然需要设置更多参数信息(除了类型之外)。 hllApiParams 中的每个元组都有一个参数“direction”(1 = 输入,2 = 输出等)、参数名称和默认值 - 详情请参阅 ctypes 文档

一旦你有了原型和参数信息,你就可以创建一个 Python“可调用”hllApi 来调用函数。您只需创建所需的变量(在我的例子中是 p1p4)并使用它们调用函数。

【讨论】:

您好,可能很久没有人在这里回复了,但我想知道一些事情来使这项工作完全适合我的程序。你能告诉我这里面的“HLLAPI”是什么吗?此外,如果我想使用函数“void BoxProperties(double L, double H, double W,double &A, double &V);”访问我的 dll您能否建议我对上述代码进行哪些更改才能使其正常工作?干杯。 hllApiParams 到底是做什么的?这些元组有什么意义?很难将本示例中的某些内容与文档相匹配。 @JonathonReinhart,我已经用底部的一些文字来充实具体示例。希望这能更好地解释它。如果没有,请告诉我,我会再试一次:-) 优秀的答案,6 年后有很大帮助:) 如果我能做到的话,我会加 100 分! 从 ctypes 新手的角度来看,这非常令人困惑。即使有解释,也不完全确定发生了什么。【参考方案5】:

也许Dispatch:

from win32com.client import Dispatch

zk = Dispatch("zkemkeeper.ZKEM") 

其中 zkemkeeper 是系统上已注册的 DLL 文件... 之后,您可以通过调用它们来访问函数:

zk.Connect_Net(IP_address, port)

【讨论】:

我正在研究这个 zkemkeeper.ZKEM,你知道我如何在 python 中获取实时事件吗?【参考方案6】:

ctypes 将是最容易使用的东西,但是(错误)使用它会使 Python 崩溃。如果你想快速做某事,而且你很小心,那就太好了。

我鼓励您查看Boost Python。是的,它要求您编写一些 C++ 代码并拥有 C++ 编译器,但您实际上并不需要学习 C++ 即可使用它,并且您可以免费获得(如啤酒)C++ compiler from Microsoft。

【讨论】:

【参考方案7】:

ctypes可以用来访问dll,这里有一个教程:

http://docs.python.org/library/ctypes.html#module-ctypes

【讨论】:

以上是关于如何使用 Python 中的 DLL 文件?的主要内容,如果未能解决你的问题,请参考以下文章

如何从 python 访问 C++ dll 中的函数,我只有一个包(一个头文件和一个 dll 文件)

Visual Studio 2010 DTE:如何使添加的 DLL 引用成为绝对引用而不是复制

MFC 应用程序中的 Qt DLL - 如何使 QDialog *真正* 模态?

如何确定 DLL 中可用的函数以在 Python 中使用

c#如何将一个类库打包成两个dll

如何使 .NET DLL 中的属性对 COM 可见(方法有效)