我目前正在进入ctypes
模块,我正在尝试使用GetWindowText
调用带有HWND
句柄的user32函数FindWindow
。这次虽然我想进一步处理并使用函数原型而不是使用ctypes.windll.user32.GetWindowText
调用函数。虽然我在将lpString
争论声明为输出参数时遇到了问题。
我的第一次尝试就是这样:
GetWindowText = cfunc("GetWindowTextA",windll.user32,c_int,
("hWnd",HWND,1),
("lpString",LPCSTR,2),
("nMaxCount",c_int,1)
)
(cfunc
是一个小包装,我发现here)
这个原型一旦被调用就会产生以下异常:
chars,name = user32.GetWindowText(handle,255)
TypeError: c_char_p 'out' parameter must be passed as default value
我认为任何输出变量必须是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
我希望有人知道如何使用GetWindowText
原型正确调用ctypes
函数。
修改
通过进一步的研究,我可以让它发挥作用,至少在某种程度上。我修复的第一个问题是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的指针,该cstring始终是255个字符长,忽略nMaxCount
指定的大小。
在我看来,我认为没有人能够使用定义为输出参数的动态大小的cstring来使用该函数。唯一的可能性似乎是没有输出参数功能并将LPCSTR
定义为输入参数。然后被调用者需要用自己的ctypes.create_string_buffer()
创建一个缓冲区并将其传递给函数(就像在C中一样)。
答案 0 :(得分:2)
您必须为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:
import win32gui
h = win32gui.FindWindow(None,'Untitled - Notepad')
print(win32gui.GetWindowText(h))
答案 1 :(得分:1)
是的。您必须为每个呼叫创建缓冲区。如果您让函数定义执行此操作,您将在以后如何访问缓冲区?
您似乎还必须告诉它期望指针c_char
与POINTER(c_char)
,而不仅仅是c_char_p
或LPSTR
。不知道为什么会发生这种情况。
无论如何,这应该有效:
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