使用重定向的stdin处理子进程中的kbhit

问题描述 投票:1回答:1

我编写了一个程序来启动另一个进程,并使用Windows API(CreateProcess,CreatePipe等)将其标准I / O重定向到管道。

该程序应启动多个不同的控制台程序,并使用stdio与它们通信。

这一切都运行良好(我可以写入进程的stdin并使用管道读取进程),直到我尝试启动并与使用kbhit的程序进行通信。

为了简化它我想要开始的程序使用标准输入:

while(1)
{
    if(kbhit())
    {
        fgets(line, sizeof(line), stdin);
        //do something with line
    }
    Sleep(100);
 } 

结果是fgets永远不会被调用,因为即使我已经写入我已经重定向stdin的管道,kbhit也不会返回true。我知道因为我调试了其他程序。我试图删除kbhit的调用,然后它确实工作,但我无法更改该代码。

有没有办法向进程发送一些东西,以便子进程中的kbhit返回true?

c++ windows winapi
1个回答
2
投票

_kbhit函数检查控制台是否有最近的击键。它从不检查标准输入,但打开CONIN$(控制台输入)并始终从这里读取。

所以在这里重定向stdin没什么可给的。如果我们与孩子共享相同的控制台,我们可以使用WriteConsoleInput完成此任务。将字符串写入子级的示例:

void write_to_conin(PCWSTR msg)
{
    if (ULONG len = (ULONG)wcslen(msg))
    {
        if (INPUT_RECORD* lpBuffer = new INPUT_RECORD[len])
        {
            INPUT_RECORD* pir = lpBuffer;
            ULONG n = len;
            do 
            {
                WCHAR UnicodeChar = *msg++;
                WORD wVirtualKeyCode = UnicodeChar;
                DWORD dwControlKeyState = CAPSLOCK_ON;

                if ((USHORT)(UnicodeChar - 'a') <= (USHORT)('z' - 'a'))
                {
                    dwControlKeyState = 0;
                    wVirtualKeyCode &= ~0x20;
                }

                pir->Event.KeyEvent.bKeyDown = TRUE;
                pir->Event.KeyEvent.dwControlKeyState = dwControlKeyState;
                pir->Event.KeyEvent.wRepeatCount = 1;
                pir->Event.KeyEvent.uChar.UnicodeChar = UnicodeChar;
                pir->Event.KeyEvent.wVirtualKeyCode = wVirtualKeyCode;
                pir->Event.KeyEvent.wVirtualScanCode = (WORD)MapVirtualKey(wVirtualKeyCode, MAPVK_VK_TO_VSC);
                pir++->EventType = KEY_EVENT;
            } while (--n);

            HANDLE hcon = CreateFileW(L"CONIN$", FILE_GENERIC_WRITE, FILE_SHARE_VALID_FLAGS, 0, OPEN_EXISTING, 0, 0);

            if (hcon != INVALID_HANDLE_VALUE) 
            { 
                WriteConsoleInput(hcon, lpBuffer, len, &n); 
                CloseHandle(hcon); 
            }

            delete [] lpBuffer;
        }
    }
}
© www.soinside.com 2019 - 2024. All rights reserved.