如果stdin被另一个进程的管道替换,Std::getline将中断
std::getline breaks if stdin is replaced by pipe of another process
我有一个简单的示例程序,它从std::cin读取并写入std::cout。如果在cmd.exe或visual studio调试器中运行,它可以正常工作。代码(server.cpp):
#include <iostream>
#include <istream>
#include <ostream>
#include <string>
int main(int argc, char* argv[])
{
std::string input;
while (std::getline(std::cin, input))
{
if (input == "dog")
{
std::cout << "cat" << std::endl;
}
else if (input == "white")
{
std::cout << "black" << std::endl;
}
else if (input == "quit")
{
std::cout << "exiting" << std::endl;
return 0;
}
else if (input != "")
{
std::cout << "unknown" << std::endl;
}
}
std::cout << "error" << std::endl;
}
现在我想在另一个进程中运行它,这个进程写入标准输入并从标准输出读取。我创建了两个管道,并使用CreateProcess启动一个进程,其中一个管道的读句柄作为StdInput句柄,另一个管道的写句柄作为stoutput句柄。代码(client.cpp):
#include <Windows.h>
#include <cassert>
#include <iostream>
#include <ostream>
#include <string>
namespace
{
class Server
{
public:
Server() :
m_pi()
{
SECURITY_ATTRIBUTES sa = {sizeof(SECURITY_ATTRIBUTES)};
sa.bInheritHandle = TRUE;
assert(CreatePipe(&m_ro, &m_wo, &sa, 0));
assert(SetHandleInformation(m_ro, HANDLE_FLAG_INHERIT, 0));
assert(CreatePipe(&m_ri, &m_wi, &sa, 0));
assert(SetHandleInformation(m_ri, HANDLE_FLAG_INHERIT, 0));
STARTUPINFO si = {sizeof(STARTUPINFO)};
si.dwFlags |= STARTF_USESTDHANDLES;
si.hStdInput = m_ri;
si.hStdOutput = m_wo;
assert(CreateProcess(L"..\Debug\server.exe", 0, 0, 0, 1, 0, 0, 0, &si, &m_pi));
}
~Server()
{
execute("quitn");
assert(WaitForSingleObject(m_pi.hProcess, INFINITE) != WAIT_FAILED);
assert(CloseHandle(m_pi.hThread));
assert(CloseHandle(m_pi.hProcess));
assert(CloseHandle(m_wi));
assert(CloseHandle(m_ri));
assert(CloseHandle(m_wo));
assert(CloseHandle(m_ro));
}
std::string execute(std::string const& cmd)
{
DWORD num_bytes;
assert(WriteFile(m_wi, cmd.c_str(), (DWORD)cmd.size(), &num_bytes, 0));
std::string output;
DWORD n = 0;
while (n == 0)
{
Sleep(0);
assert(PeekNamedPipe(m_ro, 0, 0, 0, &n, 0));
if (n > 0)
{
output.resize(n);
assert(ReadFile(m_ro, &output[0], n, &num_bytes, 0));
}
}
return output;
}
private:
HANDLE m_wo, m_ro, m_wi, m_ri;
PROCESS_INFORMATION m_pi;
};
Server g_server;
}
int main(int argc, char* argv[])
{
std::cout << g_server.execute("whiten") << std::endl;
std::cout << g_server.execute("foobarn") << std::endl;
std::cout << g_server.execute("dogn") << std::endl;
}
问题是客户端只收到消息"error",所以服务器的std::cin似乎被破坏了。
我的问题是,我做错了什么?
您正在禁用子进程将用于从stdin
读取的句柄的继承-子进程需要继承该句柄。而不是:
SetHandleInformation(m_ri, HANDLE_FLAG_INHERIT, 0);
尝试以下操作来禁用服务器进程用于写入子进程stdin
的句柄的继承:
SetHandleInformation(m_wi, HANDLE_FLAG_INHERIT, 0);
相关文章:
- 在 std::getline 和 std::cin 期间卡在循环中
- 确切地说,如何解释 std::getline(stream, string) 函数在C++中填充的字符串
- std::getline没有在while循环中重新请求用户输入
- 如何将 std::getline 转换为模板类型?
- 如何在 std::getline 中自定义分隔符
- 使用std::getline()读取一行
- std::getline with std::fstream
- 在 std::getline() 上使用任何字符作为分隔符
- C++ - 重复 std::getline() 作为用户整数输入?
- std::getline 读取最后一个字符串两次
- std::getline 未产生正确的输出 (C++)
- C STD :: getline / std :: Sort无法正常工作
- std::getline(std::cin, string) 可能因键盘输入失败的方式
- 可以将std::getline()与从std::string中移出的字符串一起使用吗
- 为什么在同一输入上调用std::getline()n次会产生第n个标记化
- std :: getline(输入,d.info)不起作用
- 用弦乐库解析行时的std :: getline问题
- std :: getline()读取马车返回 r,如何避免
- SEGFAULT与STD :: getline一起使用定界线
- progam在std :: getline AM之后停止响应,代码中是否存在错误