Python ctypes:LPCSTR[out]param的原型

2024-09-27 21:32:41 发布

您现在位置:Python中文网/ 问答频道 /正文

我目前正在进入ctypes模块,并尝试使用^{}接收到的HWND句柄调用user32函数^{}。这一次我想更进一步,使用函数原型,而不是用ctypes.windll.user32.GetWindowText调用函数。尽管我在将lpString参数声明为输出参数时遇到问题。在

我的第一次尝试是这样的:

GetWindowText = cfunc("GetWindowTextA",windll.user32,c_int,
                  ("hWnd",HWND,1),
                  ("lpString",LPCSTR,2),
                  ("nMaxCount",c_int,1)
                  )

cfunc是我找到的一个小包装器here

一旦调用此原型,就会产生以下异常:

^{pr2}$

我认为任何输出变量都必须是POINTER(...)类型,因此我将定义改为:

GetWindowText = cfunc("GetWindowTextA",windll.user32,c_int,
                      ("hWnd",HWND,1),
                      ("lpString",POINTER(c_char),2),
                      ("nMaxCount",c_int,1)
                      )

但这也产生了一个例外:

    chars,name = user32.GetWindowText(handle,255)
ctypes.ArgumentError: argument 2: <type 'exceptions.TypeError'>: wrong type

我希望有人知道如何使用ctypes原型来正确调用GetWindowText函数。在

编辑:

通过进一步的研究,我至少可以让它发挥作用。我修复的第一个问题是cfunc()的用法,它有错误的调用说明符。我定义了该函数的精确副本,并将其命名为winfunc(),并将return CFUNCTYPE(result, *atypes)((name, dll), tuple(aflags))替换为return WINFUNCTYPE(result, *atypes)((name, dll), tuple(aflags))。在

然后我进一步检查了原型。看起来,如果您将("someParameter",POINTER(aType),2)传递给WINFUNCTYPE,它将在调用时创建一个aType对象,并将指向该对象的指针传递给函数。在返回的元组中,您可以访问aType对象。这带来了另一个问题。cstring是一个字符数组;因此需要告诉ctypes创建一个c_char array。这意味着:

GetWindowText = winfunc("GetWindowTextA",windll.user32,c_int,
                  ("hWnd",HWND,1),
                  ("lpString",POINTER(c_char*255),2),
                  ("nMaxCount",c_int,1)
                  )

很好用。但不幸的是,ctypes现在将传递一个指向cstring的指针,该指针总是255个字符长,忽略了nMaxCount指定的大小。在

在我看来,我认为没有办法让这个函数与定义为输出参数的动态大小的cstring一起工作。唯一的可能性似乎是不使用输出参数特性,而将LPCSTR定义为输入参数。然后被调用者需要用ctypes.create_string_buffer()自己创建一个缓冲区,并将其传递给函数(就像在C中一样)。在


Tags: 函数参数定义ctypes原型intpointerhwnd
2条回答

是的。你必须为每个调用创建缓冲区。如果让函数定义来做,那么以后如何访问缓冲区?在

您似乎还必须告诉它期望一个带有POINTER(c_char)的指针c_char,而不仅仅是c_char_p或{}。但不知道为什么会这样。在

无论如何,这应该是有效的:

from ctypes import *
from ctypes.wintypes import *

# defs

FindWindowF = WINFUNCTYPE(HWND, LPSTR, LPSTR)
FindWindow = FindWindowF(windll.user32.FindWindowA)

GetWindowTextF = WINFUNCTYPE(c_int, HWND, POINTER(c_char), c_int)
GetWindowText = GetWindowTextF(windll.user32.GetWindowTextA)

# code

text = create_string_buffer(255)

hwnd = FindWindow(None, 'Untitled - Notepad')
GetWindowText(hwnd, text, sizeof(text))

print text.value

必须为out参数创建一个字符串缓冲区。可以将函数包装起来,使其稍微透明:

# python3
from ctypes import *

_GetWindowText = WinDLL('user32').GetWindowTextW
_GetWindowText.argtypes = [c_void_p,c_wchar_p,c_int]
_GetWindowText.restype = c_int

def GetWindowText(h):
    b = create_unicode_buffer(255)
    _GetWindowText(h,b,255)
    return b.value

FindWindow = WinDLL('user32').FindWindowW
FindWindow.argtypes = [c_wchar_p,c_wchar_p]
FindWindow.restype = c_void_p

h = FindWindow(None,'Untitled - Notepad')
print(GetWindowText(h))

或者在这种情况下,您可以使用pywin32

^{pr2}$

相关问题 更多 >

    热门问题