创建全局键盘钩子

6
我正在尝试创建一个全局键盘钩子,用于读取热键程序中的按键,但我遇到了一些问题。首先,每当我激活我的钩子时,键盘就会变得迟缓。屏幕上出现的每个字符之间几乎有1秒钟的延迟。第二个问题是,我在DLL回调函数中编写的任何代码都没有被执行。

这是我的代码:

DLL cpp文件

#include<Windows.h>
#include "keyDLL.h"

using namespace std;

LRESULT CALLBACK HookProcedure(int code, WPARAM wParam, LPARAM lParam)
{
    return CallNextHookEx(0, code, wParam, lParam);
}

DLL头文件

#include "windows.h"

extern "C" __declspec(dllexport) LRESULT CALLBACK HookProcedure(int code, WPARAM wParam, LPARAM lParam);

挂钩CPP文件

#include<Windows.h>
#include<iostream>
using namespace std;

HHOOK WINAPI SetWindowsHookEx(_In_  int idHook, _In_  HOOKPROC lpfn, _In_  HINSTANCE hMod, _In_  DWORD dwThreadId
static HINSTANCE dllHandle;
HOOKPROC address;
static HHOOK keyboard_hook;

int main()
{
    dllHandle = LoadLibrary(TEXT("keyDLL.dll"));
    address = (HOOKPROC)GetProcAddress(dllHandle, "HookProcedure");
    cout << address << endl << dllHandle << endl;
    keyboard_hook = SetWindowsHookEx(WH_KEYBOARD_LL, address, dllHandle, 0);

    system("pause");
}

这个版本的代码只是一个简单的削减版,仅设置了键盘钩子。我删除了所有其他代码,以尝试找出导致问题的原因。根据我的理解(显然不是很多),这段代码应该挂钩键盘,但不对按键进行任何操作。我不明白为什么它会让我的键盘速度变慢。我使用dumpbin确保DLL导出函数没有名称混淆。我正在运行64位的Windows 8,我的DLL和exe文件都编译为64位。当我将地址和dllHandle变量输出到控制台时,我得到了内存地址,所以我知道至少代码的一部分在工作。感谢您提供任何帮助。


3
WH_KEYBOARD_LL钩子需要一个消息循环而不是DLL。如果您的程序没有调用GetMessage(),Windows将无法调用消息钩子函数,因此程序会变得“迟缓”。 - Hans Passant
3
可能是C++控制台应用程序,SetWindowsHookEx回调从未被调用的重复问题。 - Raymond Chen
@ Hans Passant,如果我使用WH_KEYBOARD而不是WH_KEYBOARD_LL,它会更好地工作吗?或者对于全局钩子,这样做行不通?我并不完全清楚两者之间的所有区别。 - Phil Gibson
低级钩子需要一个消息循环。创建一个。 - David Heffernan
3
你应该改用 Raw Input API 而不是使用 WH_KEYBOARD_LL。它更安全、更高效。即使 LowLevelKeyboardProc 文档 也这样说:“大多数需要使用低级别钩子的应用程序,应该监视原始输入。这是因为原始输入可以比低级别钩子更有效地异步监视其他线程目标的鼠标和键盘消息。” - Remy Lebeau
1个回答

11

另一种方法是下面的代码:

#define _WIN32_WINNT 0x0400
#pragma comment( lib, "user32.lib" )

#include <iostream>
#include <windows.h>
#include <stdio.h>

HHOOK hKeyboardHook;

__declspec(dllexport) LRESULT CALLBACK KeyboardEvent (int nCode, WPARAM wParam, LPARAM lParam)
{
    DWORD SHIFT_key=0;
    DWORD CTRL_key=0;
    DWORD ALT_key=0;


    if  ((nCode == HC_ACTION) &&   ((wParam == WM_SYSKEYDOWN) ||  (wParam == WM_KEYDOWN)))      
    {
        KBDLLHOOKSTRUCT hooked_key =    *((KBDLLHOOKSTRUCT*)lParam);
        DWORD dwMsg = 1;
        dwMsg += hooked_key.scanCode << 16;
        dwMsg += hooked_key.flags << 24;
        char lpszKeyName[1024] = {0};

        int i = GetKeyNameText(dwMsg,   (lpszKeyName+1),0xFF) + 1;

        int key = hooked_key.vkCode;

        SHIFT_key = GetAsyncKeyState(VK_SHIFT);
        CTRL_key = GetAsyncKeyState(VK_CONTROL);
        ALT_key = GetAsyncKeyState(VK_MENU);

            printf("Keycode = %c\n",key);

        if (key >= 'A' && key <= 'Z')   
        {

            if  (GetAsyncKeyState(VK_SHIFT)>= 0) key +=32;

            /*********************************************
            ***   Hotkey scope                         ***
            ***   do stuff here                        ***
            **********************************************/

            if (CTRL_key !=0 && key == 'y' )
            {
               MessageBox(NULL, "CTRL-y was pressed\nLaunch your app here", "H O T K E Y", MB_OK); 
               CTRL_key=0;

               //do stuff here




            }

            if (CTRL_key !=0 && key == 'q' )
            {
                MessageBox(NULL, "Shutting down", "H O T K E Y", MB_OK); 
               PostQuitMessage(0);
            }




            SHIFT_key = 0;
            CTRL_key = 0;
            ALT_key = 0;

        }

    }
    return CallNextHookEx(hKeyboardHook,    nCode,wParam,lParam);
}

void MessageLoop()
{
    MSG message;
    while (GetMessage(&message,NULL,0,0)) 
    {
        TranslateMessage( &message );
        DispatchMessage( &message );
    }
}

DWORD WINAPI my_HotKey(LPVOID lpParm)
{
    HINSTANCE hInstance = GetModuleHandle(NULL);
    if (!hInstance) hInstance = LoadLibrary((LPCSTR) lpParm); 
    if (!hInstance) return 1;

    hKeyboardHook = SetWindowsHookEx (  WH_KEYBOARD_LL, (HOOKPROC) KeyboardEvent,   hInstance,  NULL    );
    MessageLoop();
    UnhookWindowsHookEx(hKeyboardHook);
    return 0;
}

int main(int argc, char** argv)
{
    HANDLE hThread;
    DWORD dwThread;

    printf("CTRL-y  for  H O T K E Y  \n");
    printf("CTRL-q  to quit  \n");

    hThread = CreateThread(NULL,NULL,(LPTHREAD_START_ROUTINE)   my_HotKey, (LPVOID) argv[0], NULL, &dwThread);

       /* uncomment to hide console window */
    //ShowWindow(FindWindowA("ConsoleWindowClass", NULL), false);

    if (hThread) return WaitForSingleObject(hThread,INFINITE);
    else return 1;

}

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