什么可能导致低级鼠标钩子错过按钮事件?
Posted
技术标签:
【中文标题】什么可能导致低级鼠标钩子错过按钮事件?【英文标题】:What might cause a low-level mouse hook to miss button-up events? 【发布时间】:2018-02-02 13:44:30 【问题描述】:我编写了一个简单的程序来将所有鼠标事件打印到调试视图:
#include <windows.h>
#include <stdio.h>
HHOOK g_mouseEventHook = NULL;
LRESULT CALLBACK mouseEventHookProc( int code, WPARAM wParam, LPARAM lParam )
if ( code != HC_ACTION )
return ::CallNextHookEx( g_mouseEventHook, code, wParam, lParam );
PMSLLHOOKSTRUCT mi = (PMSLLHOOKSTRUCT)lParam;
char buf[2048];
sprintf( buf, __FUNCTION__": caught event: wparam = 0x%08x, time = %d, x = %d, y = %d", wParam, mi->time, mi->pt.x, mi->pt.y );
OutputDebugStringA( buf );
return ::CallNextHookEx( g_mouseEventHook, code, wParam, lParam );
int main()
g_mouseEventHook = ::SetWindowsHookEx( WH_MOUSE_LL, mouseEventHookProc, ::GetModuleHandle( NULL ), 0 );
MSG msg;
while ( ::GetMessage( &msg, NULL, 0, 0 ) > 0 )
::TranslateMessage( &msg );
::DispatchMessage( &msg );
return 0;
代码无需清理(例如取消注册挂钩或干净地退出消息循环);您只需 Ctrl+C 即可中止它。
我使用 Visual Studio 2013 通过运行编译了这个
cl main.cpp user32.lib
运行它时,调试器输出显示(使用例如DebugView 可见)此输出表示“慢”单击,我按下鼠标左键,稍等片刻,然后松开鼠标左键:
[1660] mouseEventHookProc: caught event: wparam = 0x00000201, time = 17367092, x = 546, y = 90
[1660] mouseEventHookProc: caught event: wparam = 0x00000200, time = 17367092, x = 546, y = 90
[1660] mouseEventHookProc: caught event: wparam = 0x00000202, time = 17368636, x = 546, y = 90
[1660] mouseEventHookProc: caught event: wparam = 0x00000200, time = 17368636, x = 546, y = 90
wparam
值对应于标准窗口消息:
WM_MOUSEMOVE
0x0201 => WM_LBUTTONDOWN
0x0202 => WM_LBUTTONUP
即在上面的跟踪中,我的程序看到了鼠标向下、鼠标移动、鼠标向上、鼠标移动。似乎每个按钮事件总是伴随着一个移动事件到相同的坐标(并且具有相同的时间戳)。
当点击速度快一点时,就像我通常做的那样,'up' 事件通常会被丢弃。 IE。调试跟踪如下所示:
[1660] mouseEventHookProc: caught event: wparam = 0x00000201, time = 17371038, x = 546, y = 90
[1660] mouseEventHookProc: caught event: wparam = 0x00000200, time = 17371054, x = 546, y = 90
[1660] mouseEventHookProc: caught event: wparam = 0x00000200, time = 17371132, x = 546, y = 90
即只有一个鼠标按下,然后是两个鼠标移动事件。时间戳在所有情况下都不同,并且完全丢失了 up 事件。
我在 VMware Fusion 中运行的 Windows 7 VM 中进行这些实验。
是否有人能够重现和/或解释这种行为?我的测试程序中是否存在错误,或者这是我缺少的低级挂钩的一些特殊行为?
【问题讨论】:
代码可能不是最优的但是是正确的。猜测带有调试视图输出的东西。准确检查 - 使用 4 个全局或静态变量:static ULONG ld, lu, rd, ru;
并在每次调用 switch (wParam) case WM_LBUTTONDOWN: ld++; break; case WM_LBUTTONUP: lu++; break; case WM_RBUTTONDOWN: rd++; break; case WM_RBUTTONUP: ru++; break;
时增加对应的变量。每次都打印出来,或者在你删除钩子之后
@RbMm 有趣的想法 - 我试过了,但计数器与 DebugView 输出相匹配:除非我点击得很慢,否则“向上”事件比“向下”事件少。
在这种情况下不知道什么会产生这种情况。 i for test 现在尽可能快地点击多次 - 并且在测试结束时得到 - 相等 (ld==lu &&
rd==ru`)
刚刚在 MSVC 2017 中尝试过(从带有 F5
的 IDE 运行) - 没有“额外”移动事件 并且每个都有一个 up
事件down
,慢或快。不过,我确实做了一个微小的代码更改:在调试消息的末尾添加了一个新行 ;-)
我刚刚想到的一件事是,我在虚拟机中工作的事实可能会影响这一点(我相应地扩展了我的问题)。我会在真机上试试这个。
【参考方案1】:
我知道钩子可能会错过某些事件的唯一方法是,如果钩子列表中有其他人在你前面,并且他们通过不调用 CallNextHookEx
来吃掉该事件。
如果你的钩子函数太慢,那么 Windows 可能会在不告诉你的情况下解开你,但这意味着你不会从该 HHOOK 实例中获得更多事件。
VM 可能会在您的主机和来宾计算机之间的某处失去点击,尤其是当您在 VM 窗口和主机上的其他内容之间切换时。您可以通过创建一个小测试应用程序来排除这种情况,该应用程序还打印其鼠标事件并在客户机中运行它。
【讨论】:
以上是关于什么可能导致低级鼠标钩子错过按钮事件?的主要内容,如果未能解决你的问题,请参考以下文章