SetWindowsHookEx冻结我的键盘和鼠标,以及其他奇怪的事情发生

bob*_*y_z 1 delphi winapi

我在Delphi中编写笑话程序(发出奇怪的声音,用户使用鼠标或键盘),它使用全局钩子捕获鼠标和键盘事件.
这是负责处理这个钩子的函数:

procedure MKHOOK(code: Integer;wp: wParam;lp: lParam); stdcall;
var
hh: HHOOK;
begin
  PlaySound('fart.wav');
  CallNextHookEx(hh,code,wp,lp);
end;
Run Code Online (Sandbox Code Playgroud)

当Form创建时,Hook开始:

procedure TForm6.FormCreate(Sender: TObject);
begin
   MH := SetWindowsHookEx(WH_MOUSE_LL,@MKHOOK,hInstance,0);
   KH := SetWindowsHookEx(WH_KEYBOARD_LL,@MKHOOK,hInstance,0);
end;
Run Code Online (Sandbox Code Playgroud)


当程序结束时,MH和KH是HHOOK Hook的类型被删除的地方:

procedure TForm6.FormDestroy(Sender: TObject);
begin
   UnhookWindowsHookEx(MH);
   UnhookWindowsHookEx(KH);
end;
Run Code Online (Sandbox Code Playgroud)

问题开始,当我尝试运行它时会发生奇怪的事情:

  • 鼠标和键盘停止工作,我必须使用ctrl + alt + del来恢复它
  • 一些键在程序执行后没有工作,当我停止它时(ctrl,alt,tab)
  • 有些键会改变它们的行为,例如,当我使用箭头键时,它们会转动我的屏幕视图.

这段代码有什么问题?为什么它不起作用?对不起我的英语不好 ;)

Dav*_*nan 6

这里有一些问题.最明显的:

  • 拨打电话PlaySound太贵了.如果你PlaySound每次执行钩子都会打电话,它可能会使系统陷入困境.
  • 您的钩子签名是错误的.它们需要是返回的函数LRESULT.在两种情况下,文档中描述了所需的返回值,如下所示:如果nCode小于零,则挂钩过程必须返回CallNextHookEx返回的值.
  • 您不进行任何错误检查.始终检查API调用是否有错误.每个API调用的文档中都讨论了错误处理.

第一个参数CallNextHookEx被忽略,所以你也可以通过0.在我看来,使用单独的钩子程序更清晰.

钩子函数必须如下所示:

function MouseHook(code: Integer; wp: wParam; lp: lParam): LRESULT; stdcall;
begin
  Result := CallNextHookEx(0, code, wp, lp);
end;

function KeyboardHook(code: Integer; wp: wParam; lp: lParam): LRESULT; stdcall;
begin
  Result := CallNextHookEx(0, code, wp, lp);
end;
Run Code Online (Sandbox Code Playgroud)

显然这些钩子还没有做任何事情,但是在我们尝试运行之前让我们走一走.

安装这样的钩子:

MH := SetWindowsHookEx(WH_MOUSE_LL, @MouseHook, hInstance, 0);
if MH = 0 then
  // handle error
KH := SetWindowsHookEx(WH_KEYBOARD_LL, @KeyboardHook, hInstance, 0);
if KH = 0 then
  // handle error
Run Code Online (Sandbox Code Playgroud)

我认为很明显你没有详细阅读文档.这些API使用起来很棘手.您需要精确编码.我建议您再次仔细阅读文档.

  • "你传递一个未初始化的变量作为钩子参数" - 从技术上讲,`CallNextHookEx()`忽略第一个参数,所以你传递给它的并不重要.你可以传递0,在这种情况下*可以*对两个钩子使用单个钩子程序,但我不推荐它.两个专用程序更易于管理. (2认同)