- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在尝试重定向 Windows 中子进程的 stdout
。两者都是控制台程序。我没有子进程的源代码,所以我不能强制它刷新缓冲区。正如所讨论的here和 here ,对于 printf
和类似的实现,C 运行时缓冲除控制台和打印机之外的所有内容。因此,解决方案显然是使用 CreateConsoleScreenBuffer
来创建控制台屏幕缓冲区。我正在使用 the approach from codeproject .
PROCESS_INFORMATION pi;
HANDLE hConsole;
const COORD origin = { 0, 0 };
// Default security descriptor. Set inheritable.
SECURITY_ATTRIBUTES sa;
sa.nLength = sizeof(sa);
sa.lpSecurityDescriptor = NULL;
sa.bInheritHandle = TRUE; // So the child can use it
// Create and initialize screen buffer
hConsole = CreateConsoleScreenBuffer(
GENERIC_READ | GENERIC_WRITE, // Desired access
FILE_SHARE_WRITE | FILE_SHARE_READ, // share mode to child processes
&sa, // SECURITY_ATTRIBUTES
CONSOLE_TEXTMODE_BUFFER, // Must be this.
NULL // Reserved. Must be NULL
);
DWORD dwDummy;
FillConsoleOutputCharacter(hConsole, '\0', MAXLONG, origin, &dwDummy)
现在我将 child 的标准输出定向到控制台并启动进程
STARTUPINFO si;
ZeroMemory(&si, sizeof(STARTUPINFO));
si.cb = sizeof(STARTUPINFO);
si.dwFlags = STARTF_FORCEOFFFEEDBACK | STARTF_USESTDHANDLES; // first one prevents cursor from showing loading.
si.hStdOutput = hConsole;
//...
// Get the command line and environmental block
//...
if (! CreateProcess(
NULL, // module name.
(char*)command_line.c_str(), // command line
NULL, // process SECURITY_ATTRIBUTES
NULL, // thread SECURITY_ATTRIBUTES
TRUE, // inherit handles
NULL, // creation flags
enviros, // environmentBlock (enviros=NULL for testing)
cDir, // working directory
&si, // STARTUP_INFO object
&pi // PROCESSINFO
) ){
auto test = GetLastError();
CloseHandle(hConsole);
return false;
}
CloseHandle(pi.hThread);
然后,在循环中,我可以使用 ReadConsoleOutputCharacter
获取输出,如代码项目链接所示。看起来像
//... some initialization
GetConsoleScreenBufferInfo(hConsole, &csbi);
DWORD count = (csbi.dwCursorPosition.Y - lastpos.Y)*lineWidth + csbi.dwCursorPosition.X - lastpos.X;
LPTSTR buffer = (LPTSTR)LocalAlloc(0, count * sizeof(TCHAR));
ReadConsoleOutputCharacter(hConsole, buffer, count, lastpos, &count);
DWORD dwDummy;
FillConsoleOutputCharacter(hConsole, '\0', count, lastpos, &dwDummy);
//... Now move the cursor and grab the data from `buffer`
在 Windows 7/8.1 上,这适用于所有程序。 在 Windows 10 上,某些程序似乎绕过提供的句柄并直接打印到父控制台,这使我无法根据需要获取输出。
我还有一条线索。如果我强制进程创建一个新的控制台窗口,即
CreateProcess(NULL, (char*)command_line.c_str(), NULL, NULL, TRUE, CREATE_NEW_CONSOLE, enviros, cDir, &si, &pi)
但仍然重定向STARTUPINFO
对象中的句柄,新的控制台将显示一行The system cannot write to the specified device
,这恰好是 the MSDN docs 中 Windows 错误代码 ERROR_WRITE_FAULT = 29
的确切措辞.这只发生在那些没有按预期工作的程序上。其他程序,新控制台是空白的,它们仍然按预期运行。
我首先想到的是权限问题,但我对相关可执行文件的目录拥有广泛的权限。
我尝试过的事情
CreateProcess
中显式设置工作目录CreateProcess
的 SECURITY_ATTRIBUTES
对象CreateConsoleScreenBuffer
的 SECURITY_ATTRIBUTES
对象更深层次
感谢@PaulSanders 的建议。
为了帮助任何可能想要提供帮助的人,我提供了 a modified version of the RTConsole code from the codeproject page .它应该在 Visual Studio 中编译,只需要一个重定向,我认为。在第 135 行附近,我在采用预期路径的输出前面添加了一个小字符串。为了方便起见,我也在那里提供了一个预编译版本。
一个不起作用的软件示例是 EWBF miner .要使用我上面提供的代码进行快速测试,您可以运行
RTConsole2.exe "path\to\ewbf.exe" --help
您会看到输出中不存在前置标志。
另一方面,ccminer ,您将在运行时获得预期的行为
RTConsole2.exe "path\to\ccminer.exe" --help
最佳答案
Windows 10 中的新控制台实现有一个错误,其中高级 WriteConsole
和 WriteFile
到非事件屏幕缓冲区,而不是总是写入事件屏幕缓冲区.低级 WriteConsoleOutput[Character]
工作正常。使用旧版控制台也可以。您可以在属性对话框中启用旧版控制台。
请注意,如果由于 CREATE_NEW_CONSOLE
标志而分配新控制台,则进程无法在父控制台中为屏幕缓冲区使用继承的句柄。尝试写入屏幕缓冲区文件将失败,因为它未绑定(bind)到调用者的附加控制台(即 conhost.exe 的实例)。
绑定(bind)的控制台文件包括“CON”、“CONIN$”、“CONOUT$”和来自 CreateConsoleScreenBuffer
的新屏幕缓冲区。还有未绑定(bind)的输入和输出控制台文件,在分配新控制台时将其设置为标准句柄(例如通过 AllocConsole()
)。这些句柄访问任何连接的控制台 [*] 的输入缓冲区和事件屏幕缓冲区。请注意,一个进程可以拥有绑定(bind)到多个控制台的控制台文件的句柄,并且可以使用 AttachConsole
在控制台之间切换。
另请注意,某些程序会打开“CONOUT$”而不是写入StandardOutput
和StandardError
句柄,尤其是当它们需要控制台而不是标准句柄时是(例如管道或磁盘文件)。在这种情况下,在 STARTUPINFO
中设置 hStdOutput
是不够的。您必须通过 SetConsoleActiveScreenBuffer
暂时激活新的屏幕缓冲区。这不会影响调用者的标准句柄。它在连接的控制台中设置事件屏幕缓冲区,这就是“CONOUT$”打开的内容。子进程退出后,或者您知道子进程已经打开并写入新的屏幕缓冲区后,可以恢复之前的屏幕缓冲区。
[*] 在 Windows 8+ 中,这些控制台文件由 condrv.sys 设备驱动程序实现,并在“\Device\ConDrv”上打开。它们分别命名为“Console”、“CurrentIn”、“CurrentOut”、“ScreenBuffer”、“Input”和“Output”。控制台连接本身的文件名是“Connect”。在内部,后者作为进程 ConsoleHandle
打开,它在某些情况下由控制台 API 隐式使用(例如 GetConsoleWindow
、GetConsoleCP
和 GetConsoleTitle
).
关于c++ - 通过 Windows 10 C++ 中的控制台屏幕缓冲区重定向子进程标准输出,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51234926/
我需要检查在我的 RCP 应用程序中启动时是否加载了某些包。我知道有一个“主机 OSGi 控制台”可以显示 Eclipse IDE 中所有插件的状态,但我对这些不感兴趣。 我执行了以下步骤来获取我的应
在 pdb/ipdb 调试中,有用的 interact 命令为我提供了一个功能齐全的交互式 Python 控制台。 但是,这似乎始终是“标准”Python 控制台,即使我使用 ipdb 开始也是如此。
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
我搜索过但找不到答案:如何在运行 Emacs 时选择:文件、编辑、选项、缓冲区、工具、C++ 等下拉菜单在控制台模式下?不是终端菜单。 不,F10 不是答案。 最佳答案 如果不是 F10,那么 M-x
我正在制作一个每 20-40 秒截屏一次的 c# 控制台应用程序。 我试过到处找,但所有其他示例都没有使用控制台 这是我到目前为止所做的代码: using System; using System.D
尝试使用 terraform 控制台,新功能。 我使用 tfstate 进入我的项目并运行“terraform 控制台”。 我可以使用常规插值系统获取变量值、数据和资源。但是,模块很难破解,我无法正确
我正在尝试调试一段返回错误的 SQL。我不确定 django 或 mysql 是否处理错误,所以我想通过 django 控制台运行它。 有办法设置吗? 提前致谢。 最佳答案 manage.py dbs
你好是否可以在 JPanel 中绘制 java 控制台返回的内容?你有教程可以遵循吗?谢谢开关 最佳答案 我不记得在哪里找到这个,但我已使用我称为 TextAreaOutputStream 的类将输出
我对 Xcode 甚至编程都有点陌生。 在 Xcode 中,在我的代码中,如何显示控制台并清除屏幕? 我知道我可以使用 Xcode 首选项来完成此操作,但我想以编程方式完成此操作。 最佳答案 这对我有
我正在开发一个 C# 项目,我需要从没有 API 或 Web 服务的安全网站获取数据。我的计划是登录,访问我需要的页面,并解析 HTML 以获取记录到数据库所需的数据位。现在我正在使用控制台应用程序进
我是编程新手,正在尝试不同的在线事件以掌握它。我遇到了一个特定的问题,我想制作一个程序,用户输入一个值并打印一个特定的字符串。例如,当用户输入 0 时,将打印字符串“black”,输入 1 将打印字符
我想创建一个终端/控制台,用户可以在其中输入命令。我知道 java,但我是 xml 的新手,所以我想知道如何在文本下生成文本,如果它变得很长,它应该是可滚动的,这是一张图片: 这是我的 xml cpd
我有一个由随机生成的数字组成的 nxn 网格。我有一个标签显示 X 轴和 Y 轴的元素编号: 对于单个数字,它可以正确对齐,但是当网格大小增加时,标签会变得不成比例并且不会像这样对齐: 我想知道是否有
假设我创建了一个包含两个变量的结构。 struct mystruct{ public: string name; int age;}; class School :public mystruct{ p
我正在重写一个服务器程序,我想在其中添加一个简单的控制台输入。 目前,它只是提供数据并为它所做的每一件事打印出一两行,作为任何观看/调试的人的描述性措施。 我想要的是有一个始终位于底部的“粘性”输入栏
我必须编写启动另一个进程(GUI)的控制台应用程序。然后,使用其他应用程序或相同的选项,我必须能够停止子进程。此外,如果子进程从 GUI 关闭,则必须通知我执行最终任务(如果被杀死,则相同)。 我认为
我一直在尝试到处寻找以下问题的答案: Linux上的标准输出/控制台默认将内容保存到文件中吗? 我不想保存内容或重定向输出(我已经知道这一点),我只是想知道它是否已经通过 linux 中包含的某个默认
我正在尝试不同的事件,因为我是初学者并且想了解更多。我正在尝试在我的代码所在的同一行打印一个图案: int main() { int numOfWiggles; int count;
在我的一项小任务中,我被要求创建一个数组来存储从用户提供的输入中获取的姓名和地址,并且稍后能够从数组中删除姓名和地址。 如果能帮助我理解如何实现这一目标,我们将不胜感激,谢谢。 编辑 - 该数组将像地
如果您想在 Python shell 中查看特定模块中定义了哪些模块,一种选择是键入 dir(path.to.module)。不幸的是,这不仅列出了特定模块中定义的类或函数,还包括该模块导入的类或函数
我是一名优秀的程序员,十分优秀!