我目前正在进入ctypes
模块,我正在尝试调用 user32 函数GetWindowText http://msdn.microsoft.com/en-us/library/windows/desktop/ms633520%28v=vs.85%29.aspx with a HWND
处理我已经收到的使用FindWindow http://msdn.microsoft.com/en-us/library/windows/desktop/ms633499%28v=vs.85%29.aspx。这次我想进一步处理并使用函数原型而不是调用函数ctypes.windll.user32.GetWindowText
。虽然我在声明时遇到问题lpString
参数作为输出参数。
我的第一次尝试是这样的:
GetWindowText = cfunc("GetWindowTextA",windll.user32,c_int,
("hWnd",HWND,1),
("lpString",LPCSTR,2),
("nMaxCount",c_int,1)
)
(cfunc
是我找到的一个小包装纸here http://www.cs.unc.edu/~gb/blog/2007/02/11/ctypes-tricks/)
该原型一旦被调用就会产生以下异常:
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
原型设计。
Edit:
通过进一步的研究,我可以让它发挥作用,至少以某种方式。我解决的第一个问题是使用cfunc()
其中调用说明符错误。我定义了该函数的精确副本并将其命名winfunc()
并更换return CFUNCTYPE(result, *atypes)((name, dll), tuple(aflags))
with return WINFUNCTYPE(result, *atypes)((name, dll), tuple(aflags))
.
然后我进一步检查了原型设计。看起来如果你通过有点像("someParameter",POINTER(aType),2)
to 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 中一样)。