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

具体来说,如何在不编写任何额外的包装器c++代码将功能公开给Python的情况下实现这一点呢?

本机Python功能比使用第三方库更受欢迎。


当前回答

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


就上面所示的特定示例而言,IBM的EHLLAPI是一个相当一致的接口。

根据IBM文档,所有调用都传递四个void指针(EHLLAPI通过第四个参数返回返回代码,这是一个指向int的指针,因此虽然我指定int作为返回类型,但可以安全地忽略它)。换句话说,函数的C变体是:

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

这使得一个简单的ctypes函数能够做EHLLAPI库提供的任何事情,但其他库可能需要为每个库函数设置单独的ctypes函数。

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

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

其他回答

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

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

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

我鼓励你去看看Boost Python。是的,它要求你写一些c++代码,并有一个c++编译器,但你实际上不需要学习c++来使用它,你可以从微软得到一个免费的(就像啤酒一样)c++编译器。

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


就上面所示的特定示例而言,IBM的EHLLAPI是一个相当一致的接口。

根据IBM文档,所有调用都传递四个void指针(EHLLAPI通过第四个参数返回返回代码,这是一个指向int的指针,因此虽然我指定int作为返回类型,但可以安全地忽略它)。换句话说,函数的C变体是:

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

这使得一个简单的ctypes函数能够做EHLLAPI库提供的任何事情,但其他库可能需要为每个库函数设置单独的ctypes函数。

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

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

也许与调度:

from win32com.client import Dispatch

zk = Dispatch("zkemkeeper.ZKEM") 

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

zk.Connect_Net(IP_address, port)

如果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中的类实例化一个对象,并访问其中的方法。