不会进入控制台应用程序上的 Windows GetMessage 循环

does not go inside the Windows GetMessage loop on console application

本文关键字:程序上 Windows GetMessage 循环 应用程序 应用 控制台      更新时间:2023-10-16

我想检测C++中的按键,我需要使用Windows系统调用。所以,我做了一些研究,这就是我使用钩子和消息得到的:

#include <Windows.h>
#include <iostream>
#include <sstream>
#include <fstream>
#include <ctime>
using namespace std;
LRESULT CALLBACK LowLevelKeyboardProc(int code, WPARAM wParam, LPARAM lParam) {
    if (code == HC_ACTION) {
        switch (wParam) {
        case WM_KEYDOWN:
            PKBDLLHOOKSTRUCT p = (PKBDLLHOOKSTRUCT)lParam;
            char c = char(MapVirtualKey(p->vkCode, MAPVK_VK_TO_CHAR));
            cout << c << endl;
        }
    }
    return CallNextHookEx(NULL, code, wParam, lParam);
}
int main() {    
    HHOOK HKeyboard = SetWindowsHookEx(WH_KEYBOARD_LL, LowLevelKeyboardProc, 0, 0);
    MSG msg;
    BOOL bRet;
    while ((bRet = GetMessage(&msg, NULL, 0, 0)) > 0) {
        cout << "bRet = " << bRet << endl; // I want to do something here, but the program doesn't seem to go in here
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }
    UnhookWindowsHookEx(HKeyboard);
    return 0;
}
我的

问题是为什么我的程序没有进入循环(而是卡在 GetMessage 函数上)?我需要它来设置条件以在几秒钟后终止,那么我应该在哪里放置条件?我知道 GetMessage 函数读取消息,但是当我按下键盘上的键时,它仍然无法进入,并且回调函数工作正常。

事件将发布到活动窗口。控制台窗口由控制台子系统 csrss.exe 拥有,它接收事件,然后将它们转换为字符并将它们放入控制台对象中,这是应用程序的stdin

如果要以 Win32 GUI 方式处理事件,则应使用 Win32 窗口(例如 RegisterClassCreateWindow ),而不是控制台窗口。

如果您只希望回调在一段时间内工作,则可以使用可警报的等待,例如 SleepExMsgWaitForMultipleObjects ,它们接受超时。

这并不奇怪,您的线程队列中实际上没有任何消息,并且您没有窗口,因此也没有窗口队列。

如果要使用从钩子返回的值,请将代码放在钩子回调函数中。

我应该警告您,像这样的钩子在控制台应用程序中不起作用,因为控制台窗口驻留在另一个进程中。 此外,如果您查看 MSDN 页面以获取SetWindowsHookEx,您会发现 WH_KEYBOARD_LL 只是一个全局钩子,即您必须将钩子处理程序放在 DLL 库中并将钩子注入其他应用程序中。然后,您必须自己处理32/64位问题。

最后一点,当您在处理程序中说cout << c;时,那将打印在进程的输出文件句柄中,而不是您自己的句柄中。