2010-12-14 120 views
4

我有一個程序利用第三方命令行工具來生成日誌。第三方將其所有輸出生成到STDOUT,然後用戶必須使用" > test.txt"命令來保存文件。如何將STDOUT輸出保存到文本文件?

但是程序以某種方式產生了一定數量的報告,但不是整個報告。這通過使用命令行控制檯其中工程上和其上僅部分工作的程序的命令

C:\Test\ftk\ripxp>ripxp.exe -r C:\test\ftk\ntuser.dat -d "C:\System Volume\_rest ore{BB12863B-2C77-46C9-BCDA-1810E52F1089}" -p runmru > C:\test\test05.txt

是測試。

錯誤被縮小爲參數錯誤或文件保存錯誤部分(streamReader)。因此,錯誤可能是由於STDOUT沒有正確保存。

因此,可能有人請告訴代碼?謝謝!

的第三方工具(2008 H.卡維)的參數:

RipXP v.20081001 - CLI RegRipper tool 
RipXP [-r Reg hive file] [-p plugin module][-d RP dir][-lgh] 
Parse Windows Registry files, using either a single module from the plugins folder. 
Then parse all corresponding hive files from the XP Restore Points (extracted from 
image) using the same plugin. 

-r Reg hive file...Registry hive file to parse 
-g ................Guess the hive file (experimental) 
-d RP directory....Path to the Restore Point directory 
-p plugin module...use only this module 
-l ................list all plugins 
-h.................Help (print this information) 

Ex: C:\>rip -g 
C:\>rip -r d:\cases\ntuser.dat -d d:\cases\svi -p userassist 

All output goes to STDOUT; use redirection (ie, > or >>) to output to a file. 

copyright 2008 H. Carvey 

的代碼:

static void Main(string[] args) 
    { 
     // Automatically finds folder that starts with _restore 
     DirectoryInfo directoryInfo = new DirectoryInfo(@"C:\System Volume\"); 
     DirectoryInfo restoreFolder = directoryInfo.GetDirectories().FirstOrDefault(d => d.Name.StartsWith("_restore")); 

     // Gets the folder name 
     String baka = restoreFolder.Name; 

     if (restoreFolder == null) 
      throw new DirectoryNotFoundException(); 

      Process process = new Process(); 
      process.StartInfo.FileName = @"C:\test\ftk\ripxp\ripxp.exe"; 
      process.StartInfo.Arguments = @"-r C:\test\ftk\ntuser.dat -d C:\System Volume\" + restoreFolder.Name + " -p runmru"; 
      process.StartInfo.CreateNoWindow = false; 
      process.StartInfo.UseShellExecute = false; 
      process.StartInfo.RedirectStandardOutput = true; 
      process.StartInfo.RedirectStandardInput = true; 
      process.StartInfo.RedirectStandardError = true; 
      process.Start(); 

      String text = @"-r C:\test\ftk\ntuser.dat -d C:\System Volume\" +restoreFolder.Name + " -p runmru"; 
      Console.WriteLine(baka); 
      Console.WriteLine(text); 

      // Strangely the program only outputs the first section "-r C:\test\ftk\ntuser.dat" argument results..... 
      System.IO.StreamReader reader = process.StandardOutput; 
      String sRes = reader.ReadToEnd(); 
      StreamWriter SW; 
      SW = File.CreateText(@"C:\test\test01.txt"); 
      SW.WriteLine(sRes); 
      SW.Close(); 
      Console.WriteLine("File Created Successfully"); 
      reader.Close(); 

    } 
+0

如果加上'Console.Out.Flush() ;'Console.WriteLine(text);'後面,你能看到整行嗎? – SwDevMan81 2010-12-14 15:02:34

+0

如果另一個Console.WriteLine(文本);放在Console.Out.Flush()之後;那麼可以看到文字。 – JavaNoob 2010-12-14 16:07:01

回答

3

您可以讓.NET在進程退出後通知您,然後讀取輸出。因爲,正如他的回答+評論中提到的那樣,如果您在完成之前致電reader.ReadToEnd(),您將無法獲得所有輸出。如果你想到它,這很明顯 - 數據還沒有產生,所以你如何期望讀者讀它?

通過使用事件,您不會阻止啓動線程的方法,如果您有GUI應用程序並且不希望在子進程運行時凍結用戶界面,該方法可能非常有用。

// tell it to raise events and hook up a callback to when it completes 
process.EnableRaisingEvents = true; 
process.Exited += process_Exited; 
process.Start(); 

當進程完成後,該方法將被調用:

void process_Exited(object sender, EventArgs e) 
{ 
    var process = (Process)sender; 
    using (var f = File.CreateText(@"...")) 
    { 
     f.WriteLine(process.StandardOutput.ReadToEnd()); 
    } 
} 
+0

感謝您解釋隊友! – JavaNoob 2010-12-15 04:20:01

+0

*注意*:可能需要重定向StandardError !!! – Andrew 2016-03-03 05:52:38

2

的System.Console類有一個SetOut方法應該做你需要什麼。

您應該在程序執行開始時調用Console.SetOut(yourTextWriter)

就在程序的第一行補充一點:

Console.SetOut(File.CreateText(locationToSaveLogs)); 
+0

你能告訴我在哪裏添加代碼?謝謝! – JavaNoob 2010-12-14 14:59:04

+0

@Java,看我的編輯。 – jjnguy 2010-12-14 15:04:00

+1

重定向您自己的輸出,而不是您啓動的進程的輸出。 – 2010-12-14 15:24:11

5

你等待子進程結束?看起來你很早就開始閱讀它的輸出。它可以這樣做:

process.Start(); 
process.WaitForExit(); 

此外,您可以開始通過代表接收輸出結束之前。像這樣(每個文本行都需要代理):

process.OutputDataReceived += delegate(object sender, DataReceivedEventArgs e) 
{ 
    // Store e.Data somewhere. 
}; 
+0

即使沒有完成子進程,輸出應該能夠完全保存爲它的唯一一個5kb文件。 – JavaNoob 2010-12-14 15:00:31

+1

它無關緊要。當你過早地調用'reader.ReadToEnd()'時,將只有整個輸出的一小部分。 – detunized 2010-12-14 15:04:36

+0

代表另一個答案! – JavaNoob 2010-12-15 04:20:19

0

您需要使用OutputDataReceived事件。這MSDN文章介紹瞭如何做到這一點:

https://msdn.microsoft.com/en-us/library/system.diagnostics.process.outputdatareceived(v=vs.110).aspx?cs-save-lang=1&cs-lang=csharp#code-snippet-2

裏面的處理程序使用的StreamWriter喜歡寫輸出到文件:

StreamWriter sw = new StreamWriter(sOutputFilePath); 
    Process process = new Process(); 
    process.StartInfo.FileName = "ipconfig.exe"; 
    process.StartInfo.UseShellExecute = false; 
    process.StartInfo.RedirectStandardOutput = true; 
    process.OutputDataReceived += new DataReceivedEventHandler((sender, e) => 
    { 
     if (!String.IsNullOrEmpty(e.Data)) 
     { 
      sw.WriteLine(e.Data); 
     } 
    }); 

    process.Start(); 
    process.BeginOutputReadLine(); 
    process.WaitForExit(); 
    sw.Close();