gpt4 book ai didi

c# - 如何抓取父进程标准输出?

转载 作者:行者123 更新时间:2023-11-30 19:35:12 24 4
gpt4 key购买 nike

我正在编写一个实用程序 ( http://reg2run.sf.net ),以防不带参数的执行作为 Windows 应用程序(显示 OpenFileDialog 等),否则 - 作为控制台应用程序。

所以,在第一种情况下,我不想显示控制台窗口,这就是项目是 Windows 应用程序的原因。但在第二 - 我需要展示它,它是用

创建的
if (ptrNew == IntPtr.Zero)
{
ptrNew = GetStdHandle(-11);
}
if (!AllocConsole())
{
throw new ExternalCallException("AllocConsole");
}
ptrNew = CreateFile("CONOUT$", 0x40000000, 2, IntPtr.Zero, 3, 0, IntPtr.Zero);
if (!SetStdHandle(-11, ptrNew))
{
throw new ExternalCallException("SetStdHandle");
}
StreamWriter newOut = new StreamWriter(Console.OpenStandardOutput());
newOut.AutoFlush = true;
Console.SetOut(newOut);
Console.SetError(newOut);

我想要的是获取父进程标准输出并使用它(如果存在)(以防通过 cmd.exe 或 Far Manager 执行)。我该怎么做?

我试过了

static Process GetParentProc()
{
int pidParent = 0;
int pidCurrent = Process.GetCurrentProcess().Id;

IntPtr hSnapshot = CreateToolhelp32Snapshot(2, 0);
if (hSnapshot == IntPtr.Zero)
{
return null;
}

PROCESSENTRY32 oProcInfo = new PROCESSENTRY32();
oProcInfo.dwSize = (uint)Marshal.SizeOf(typeof(PROCESSENTRY32));

if (!Process32First(hSnapshot, ref oProcInfo))
{
return null;
}
do
{
if (pidCurrent == oProcInfo.th32ProcessID)
{
pidParent = (int)oProcInfo.th32ParentProcessID;
}
}
while (pidParent == 0 && Process32Next(hSnapshot, ref oProcInfo));

if (pidParent > 0)
{
return Process.GetProcessById(pidParent);
}
else
{
return null;
}

StreamWriter newOut = GetParentProc().StandardInput;

但得到 InvalidOperationException:StandardIn 没有被重定向。因为

GetParentProc().StartInfo.RedirectStandardOutput = false

最佳答案

对于需要根据上下文在 Windows 上选择是充当控制台还是 GUI 应用程序的应用程序,有几种方法:

  1. 有两个独立的应用程序,并有条件地启动另一个应用程序。
  2. 上述策略的一种变体,有两个应用程序,一个称为“app.com”(即只需重命名一个带有 COM 扩展名的控制台 EXE),另一个称为“app.exe”,这样命令行调用就会发现首先是 app.com。由于古老的 DOS 兼容性,.COM 可执行文件在 .EXE 之前被发现。 (这在 Windows 中是可配置的;请参阅 PATHEXT 环境变量。)
  3. rxvt/Cygwin 技术,这是我在其他任何地方都没有真正看到过记录的技术。

让我详细介绍一下 Cygwin 上的 rxvt 是如何工作的。 Rxvt 是一个终端仿真器,通常在 X Window 系统上运行。由于 Win32 控制台的局限性,Cygwin 将其打包为功能更全面的控制台,支持诸如大量历史行、动态调整大小、每个实例可配置的字体和颜色主题、非应用程序卡住鼠标选择等功能和复制等。为了在 Windows 上本地运行,Cygwin 附带的 rxvt 包括一个用于 Win32 的小型 X11 包装器库。出于与现有 native Win32 可执行文件兼容的原因,Windows 上的 Rxvt 实际上是一个控制台应用程序,但大多数时候您永远看不到控制台;您只会看到 rxvt 终端仿真器窗口本身。

它的工作方式在 rxvt 源代码树的 rxvt/W11/wrap/wrap.c 中,在名为 hideConsole() 的函数中具体实现。基本上,它打开它的控制台(使用 CreateFile("CONOUT$"...)),并检查光标位置是否在 (0,0)(使用 GetConsoleScreenBufferInfo () 在控制台句柄上)。

如果是,则它推断它是作为独立应用程序启动的,而不是从控制台父应用程序启动的,因此它知道操作系统已经为该进程创建了专用的 Win32 控制台。它继续隐藏这个控制台窗口,但它必须首先找到它。它使用 SetConsoleTitle 将控制台窗口的标题设置为基于应用程序名称和当前线程 ID 的唯一值。然后它使用 FindWindow 找到这个窗口的句柄(如果需要,周期性地 Sleep 几毫秒来改变标题,因为控制台窗口实际上是由不同的进程控制的完全在 Windows 中)。当它最终找到窗口句柄时,它使用 ShowWindowAsync 隐藏它,并传入 SW_HIDE

使用这种方法,您可以编写一个应用程序:

  • 如果从控制台父级启动,它可以继续使用此控制台
  • 如果作为应用程序启动,它可以选择是否隐藏控制台

唯一的缺点是在应用程序启动时控制台窗口会非常短暂地闪烁。

关于c# - 如何抓取父进程标准输出?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/377911/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com