- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我有一个长期运行的基于控制台的应用程序 Sender,它使用非缓冲输出(例如 cout << "Message"<< flush())将简单文本发送到 STDOUT。我想创建一个基于 MFC 对话框的应用程序(名为 Receiver)来启动 Sender 并可以读取它的输出。 Receiver 还应该能够检测到 Sender 何时死亡,或者如果愿意,可以杀死 Sender。发送方对接收方一无所知,我无法更改发送方的代码。
我问了一个separate question关于执行此操作的最佳方法。我的第一次尝试是为子进程创建具有重定向 STDIN 和 STDOUT 的管道,并使用异步 ReadFileEx 调用来读取 Sender 的数据。这无法正常工作,因为 ReadFileEx 函数仅触发一次,并且仅传输零字节,即使我知道 Sender
正在发送数据也是如此。
我正在使用重定向的 STDIN 和 STDOUT 创建 2 个管道,ala this MS example :
// allow the child process to inherit handles
SECURITY_ATTRIBUTES sa = {0};
sa.nLength = sizeof(sa);
sa.bInheritHandle = 1;
// create pipes with rerouted stdin & stdout
CreatePipe(&handles[h_Child_StdOut_Read], &handles[h_Child_StdOut_Write], &sa, 0);
SetHandleInformation(handles[h_Child_StdOut_Read], HANDLE_FLAG_INHERIT, 0);
CreatePipe(&handles[h_Child_StdIn_Read], &handles[h_Child_StdIn_Write], &sa, 0);
SetHandleInformation(handles[h_Child_StdIn_Read], HANDLE_FLAG_INHERIT, 0);
...Receiver
然后继续通过 CreateProcess() 启动 Sender
:
// create child process
PROCESS_INFORMATION pi = {0};
STARTUPINFO si = {0};
si.cb = sizeof(si);
si.hStdOutput = handles[h_Child_StdOut_Write];
si.hStdInput = handles[h_Child_StdIn_Read];
si.dwFlags |= STARTF_USESTDHANDLES;
CreateProcess( 0, "Sender.EXE", 0, 0, 1, 0, 0, 0, &si, &pi);
handles[h_Child_Process] = pi.hProcess;
handles[h_Child_Thread] = pi.hThread;
我的主循环基于 WaitForObjectsEx,置于可警告等待状态以支持异步文件读取。我正在等待两个句柄:一个在 Sender
过早死亡时发出信号,另一个在 Receiver
的主线程希望 Sender
死亡时发出信号.在开始循环之前,我在 Sender
的 STDOUT 上启动了重叠(异步)文件读取操作。忽略明显的内存泄漏和其他黑客行为——这是说明性的:
vector<HANDLE> wait_handles;
wait_handles.push_back(handles[h_Die_Sig]);
wait_handles.push_back(handles[h_Child_Process]);
for( bool cont = true; cont; )
{
IO* io = new IO;
memset(io, 0, sizeof(IO));
io->buf_size_ = 16 * 1024;
io->buf_ = new char[io->buf_size_];
memset(io->buf_, 0, io->buf_size_);
io->thread_ = ¶m;
io->file_ = handles[h_Child_StdOut_Read];
if( !ReadFileEx(io->file_, io->buf_, io->buf_size_, io, OnFileRead) )
{
DWORD err = GetLastError();
string err_msg = util::strprintwinerr(err);
}
DWORD rc = WaitForMultipleObjectsEx(wait_handles.size(), &wait_handles[0], FALSE, INFINITE, TRUE);
// ...
}
上面的 IO
对象是从 OVERLAPPED
公开派生的:
struct IO : public OVERLAPPED
{
char* buf_;
DWORD buf_size_;
DWORD read_;
ThreadParam* thread_;
HANDLE file_;
};
当重叠读取函数完成时,我读取传入的数据并生成一个字符串:
void CALLBACK OnFileRead(DWORD err, DWORD bytes, OVERLAPPED* ovr)
{
IO* io = static_cast<IO*>(ovr);
string msg(io->buf_, bytes);
}
Sender
对 Receiver
一无所知,它使用非常简单但非缓冲的方式将文本发送到控制台。
问题:我知道 Sender
正在将数据发送到它的 STDOUT,但是我的 OnFileRead
函数只被调用一次,而且只传输了零字节。
为什么我不能以这种方式接收Sender
的输出?我有错误,还是我做错了什么?
最佳答案
除了@DyP 指出的错误之外,您还假设 CreatePipe
在重叠模式下打开了句柄。 您的假设不正确。微软documents it :
Asynchronous (overlapped) read and write operations are not supported by anonymous pipes. This means that you cannot use the ReadFileEx and WriteFileEx functions with anonymous pipes. In addition, the lpOverlapped parameter of ReadFile and WriteFile is ignored when these functions are used with anonymous pipes.
(实际上,如果您查看 kernel32.dll
,例如在 Windows XP 上,CreatePipe
不会将第七个参数的低位设置为 NtCreateNamedPipeFile
;当使用 FILE_FLAG_OVERLAPPED
调用 CreateNamedPipe
时,该位被设置。)
寻找 Dave Hart 的 MyCreatePipeEx
实现;当需要重叠 I/O 时,它可以用作 CreatePipe
的直接替代品。只需将 PipeSerialNumber++
更改为 InterlockedIncrement(&PipeSerialNumber)
即可避免 MT 代码中的竞争条件。
关于c++ - 子进程重定向的 STDOUT 上的重叠 ReadFileEx 永远不会触发,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3661106/
我正在尝试使用谷歌浏览器的 Trace Event Profiling Tool分析我正在运行的 Node.js 应用程序。选择点样本后,我可以在三种 View 之间进行选择: 自上而下(树) 自上而
对于一个可能是菜鸟的问题,我们深表歉意,但尽管在 SO 上研究了大量教程和其他问题,但仍找不到答案。 我想做的很简单:显示一个包含大量数据库存储字符串的 Android ListView。我所说的“很
我已经开始了一个新元素的工作,并决定给 Foundation 5 一个 bash,看看它是什么样的。在创建带有水平字段的表单时,我在文档中注意到的第一件事是它们使用大量 div 来设置样式。所以我在下
我有一个 Windows 窗体用户控件,其中包含一个使用 BeginInvoke 委托(delegate)调用从单独线程更新的第 3 方图像显示控件。 在繁重的 CPU 负载下,UI 会锁定。当我附加
我有一堆严重依赖dom元素的JS代码。我目前使用的测试解决方案依赖于 Selenium ,但 AFAIK 无法正确评估 js 错误(addScript 错误不会导致您的测试失败,而 getEval 会
我正在制作一款基于滚动 2D map /图 block 的游戏。每个图 block (存储为图 block [21][11] - 每个 map 总共 231 个图 block )最多可以包含 21 个
考虑到以下情况,我是前端初学者: 某个 HTML 页面应该包含一个沉重的图像(例如 - 动画 gif),但我不想强制客户缓慢地等待它完全下载才能享受一个漂亮的页面,而是我更愿意给他看一个轻量级图像(例
我正在设计一个小软件,其中包括: 在互联网上获取资源, 一些用户交互(资源的快速编辑), 一些处理。 我想使用许多资源(它们都列在列表中)来这样做。每个都独立于其他。由于编辑部分很累,我想让用户(可能
我想比较两个理论场景。为了问题的目的,我简化了案例。但基本上它是您典型的生产者消费者场景。 (我关注的是消费者)。 我有一个很大的Queue dataQueue我必须将其传输给多个客户端。 那么让我们
我有一个二元分类问题,标签 0 和 1(少数)存在巨大不平衡。由于测试集带有标签 1 的行太少,因此我将训练测试设置为至少 70-30 或 60-40,因此仍然有重要的观察结果。由于我没有过多地衡量准
我是一名优秀的程序员,十分优秀!