C++ Win32键盘事件

7

我正在为个人兴趣工作的按键记录器,并在昨天提出了一个相关问题; While loop using a lot of CPU.

这个程序的问题是它使用了太多的CPU资源,有些人建议将输入改为基于事件而非轮询。

由于我对Win32 API不熟悉,所以我尝试寻找参考和教程,告诉我如何创建基于事件而非轮询的键盘输入。但问题是我找不到任何实例或参考资料,因为对于一个完全的新手来说,这相当难以理解。

大部分的参考资料都提到基于事件的编程驻留在GUI应用程序中,但我希望这个按键记录器应用程序是一个控制台应用程序。

从所有这些中,我的两个主要问题是:

  • 我能否使用Win32 API编写基于事件的控制台按键记录器?如果不能,我的选择是什么?

  • 是否有任何参考网站可以帮助我理解如何基于事件地捕获按键事件?

如果需要其他信息,我正在Windows XP下使用Code Blocks和GCC编译器。

2个回答

8

键盘记录器应用程序使用诸如Win32 Hooks之类的机制。具体来说,您需要设置一个WH_KEYBOARD钩子。

还有更高级的技术,比如创建自己的键盘驱动程序,但钩子是一个很好的选择。

编辑:为了让您了解钩子过程的样子,我在这里发布了我的个人实用程序的片段。

// ...
thehook = SetWindowsHookEx( WH_KEYBOARD_LL, hook_proc, hwnd, 0 );
// ...

/**
 *
 *  wParam, one of the: WM_KEYDOWN, WM_KEYUP, WM_SYSKEYDOWN, or WM_SYSKEYUP
    lParam: pointer to a KBDLLHOOKSTRUCT structure

    (*) "The hook procedure should process a message in less time than the
    data entry specified in the LowLevelHooksTimeout value in the following registry key: 
    HKEY_CURRENT_USER\Control Panel\Desktop 

    The value is in milliseconds. If the hook procedure does not 
    return during this interval, the system will pass the message to the next hook."

 *
 */
LRESULT CALLBACK
hook_proc( int code, WPARAM wParam, LPARAM lParam )
{
  static long ctrl_cnt = 0;
  static bool mmode = false;
  static DWORD time;

  KBDLLHOOKSTRUCT*  kbd = (KBDLLHOOKSTRUCT*)lParam;

  if (  code < 0
  ||   (kbd->flags & 0x10) // ignore injected events
     ) return CallNextHookEx( thehook, code, wParam, lParam );

  long ret = 1; // by default I swallow the keys
  if (  mmode  ) { // macro mode is ON
    if (  WM_KEYDOWN == wParam  )
      PostMessage(mainwnd, WM_MCR_ACCUM, kbd->vkCode, 0);

    if (  WM_KEYUP == wParam  )
      switch (kbd->vkCode) {
        case VK_ESCAPE:
          mmode = false;
          keys.removeall();
          PostMessage(mainwnd, WM_MCR_HIDE, 0, 0);
          break;

        case VK_RETURN:
          PostMessage(mainwnd, WM_MCR_EXEC, 0, 0);
          break;

        case VK_LCONTROL:
          mmode = false;
          PostMessage(mainwnd, WM_MCR_HIDE, 0, 0);
          PostMessage(mainwnd, WM_MCR_EXEC, 0, 0);
          break;
      }

    /* Which non printable keys allow passing? */
    switch( kbd->vkCode ) {
      case VK_LCONTROL:
      case VK_CAPITAL:
      case VK_LSHIFT:
      case VK_RSHIFT:
        ret = CallNextHookEx( thehook, code, wParam, lParam );
    }
  }
  else { // macro mode is OFF
    /* Ctrl pressed */
    if (  kbd->vkCode == VK_LCONTROL && WM_KEYDOWN == wParam  ) {
      ctrl_cnt = 1;
      time = kbd->time;
    }

    /* Prevent ctrl combinations to activate macro mode */
    if (  kbd->vkCode != VK_LCONTROL  )
      ctrl_cnt = 0;

    /* Ctrl released */
    if (  ctrl_cnt == 1 && WM_KEYUP == wParam  ) {
      if (  kbd->time - time > 40  ) {
        mmode = true;
        PostMessage(mainwnd, WM_MCR_SHOW, 0, 0);
      }
    }

    ret = CallNextHookEx( thehook, code, wParam, lParam ); // let it pass
  }

  return ret;
}

如何获取另一个程序的 HINSTANCE? - Matouš Vrba
@MatoušVrba,您不需要获取其他程序的句柄。 - Nick Dandoulakis
Win32 Hooks的链接已经失效。 - Steve Smith

4
请看一下SetWindowsHookEx API。
您可以在控制台应用程序和窗口应用程序中执行此操作。您需要将键盘钩子放入DLL中,以便您可以捕获自己以外的进程中的按键。
这里有一些示例代码

只是好奇,挂钩到底是做什么的?我读过它会注入到另一个程序中,或者这可能是我的理解。此外,这是否会减少按键记录器的 CPU 使用量? - CuriousUser
1
钩子是在选择的事件发生时向您的函数添加回调的一种方式。您不需要循环不断地轮询键盘。只有在必要时,您的代码才会执行。 - Ferruccio
同时请参阅此讨论https://dev59.com/rXA75IYBdhLWcg3wubun#ncIJoYgBc1ULPQZF2hlI(我认为exe可以处理*_LL消息,但其他任何操作均需要dll)。 - rogerdpack
我提交了一份编辑,将您的评论添加到您的答案中。如果您不喜欢这个编辑,请随意拒绝。 :) - Cypher

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接