C# 从控制台解码文本

C# 从控制台解码文本,c#,.net,console,decode,C#,.net,Console,Decode,我尝试执行以下代码: private void Test(object sender, RoutedEventArgs e) { ProcessStartInfo start = new ProcessStartInfo("cmd", "/c \"wbadmin start recovert -version:02/26/2014-17:38 -itemtype:File - items:C:\test\""); int exitCode;

我尝试执行以下代码:

private void Test(object sender, RoutedEventArgs e)
    {
        ProcessStartInfo start = new ProcessStartInfo("cmd",
    "/c \"wbadmin start recovert -version:02/26/2014-17:38 -itemtype:File - items:C:\test\"");

        int exitCode;
        using (Process proc = Process.Start(start))
        {
            proc.ErrorDataReceived += cmd_Error;
            proc.OutputDataReceived += cmd_DataReceived;
            proc.WaitForExit();

            exitCode = proc.ExitCode;
        }
    }

    private void cmd_DataReceived(object sender, DataReceivedEventArgs e)
    {
        if (e.Data == null) return;

        var source = Encoding.Unicode;
        var target = Encoding.UTF8;

        var sBytes = source.GetBytes(e.Data);
        var tBytes = Encoding.Convert(source, target, sBytes);

        var tString = Encoding.UTF8.GetString(tBytes);
        Console.WriteLine(tString);
    }
但是我得到了这个字符串:“wbadmin 1.0-®≠·‚‡„¨•≠‚ ™Æ¨†≠§≠Æ© ·‚‡Æ™® †‡Â®¢†Ê®®"
如何解码此字符串?

解析cmd的输出可能有点棘手,因为cmd有自己的代码页,通常等于系统的默认语言环境(您可以手动更改,例如使用命令)

详情请阅读

重定向输出时,我的工作方式(经过测试,也使用了
wbadmin
)如下:

  • 获取系统的默认区域设置:

    [DllImport("kernel32.dll")]
    public static extern int GetSystemDefaultLCID();
    
    private static int GetCmdCodePage()
    {
        int lcid = GetSystemDefaultLCID();
        var ci = System.Globalization.CultureInfo.GetCultureInfo(lcid);
        return ci.TextInfo.OEMCodePage;
    }
    
  • 获取相应的编码:

        Encoding enc = null;
        try
        {
            enc = Encoding.GetEncoding(GetCmdCodePage());
        }
        catch (Exception)
        {
            enc = Encoding.GetEncoding(855); // the value for Cyrillic
        }
    
  • 设置进程的编码:

        if (!File.Exists(Path.Combine(Environment.SystemDirectory, @"wbadmin.exe")))
        {
            Console.WriteLine("wbadmin.exe not found");
            return;
        }
        Process pr = new Process();
        ProcessStartInfo psi = new ProcessStartInfo(@"wbadmin.exe");
        psi.WindowStyle = ProcessWindowStyle.Hidden;
        psi.CreateNoWindow = true;
        psi.UseShellExecute = false;
        psi.Arguments = "/?"; // prints avaliable commands
        psi.RedirectStandardOutput = true;
        psi.RedirectStandardError = true;
        psi.Verb = "runas";
        psi.StandardOutputEncoding = enc;
        psi.StandardErrorEncoding = enc;
        pr.StartInfo = psi;
        pr.Start();
    
        pr.WaitForExit(1000);
        string error = pr.StandardError.ReadToEnd();
    
        if (!string.IsNullOrEmpty(error))
        {
            Console.WriteLine("error: " + error);
            pr.Close();
            pr.Dispose();
            return;
        }
    
        string output = pr.StandardOutput.ReadToEnd();
    
        pr.Close();
        pr.Dispose();
    

  • 你的代码看起来完全正确,但毫无意义。事实上,不管发生什么情况,C#string始终是UTF-16。cmd_DataReceived方法将UTF-16转换为字节数组,其中包含原始字符串的UTF-8表示形式,然后通过调用Encoding.UTF8.GetString(tBytes)将其转换回UTF-16

    看起来外部程序以未知编码(UTF-8?)将某些内容写入控制台,但cmd_DataReceived接收到的数据已经解码为UTF-16

    我假设,如果您确实想要将字符串从UTF-8转换为UTF-16,那么您的代码应该如下所示

    private void cmd_DataReceived(object sender, DataReceivedEventArgs e)
        {
            if (e.Data == null) return;
    
            var source = Encoding.Unicode;
            var target = Encoding.UTF8;
    
            var sBytes = source.GetBytes(e.Data);
    
            var tString = Encoding.UTF8.GetString(sBytes);
            Console.WriteLine(tString);
        }
    

    你能试试
    source=Encoding.Utf8
    source=Encoding.GetEncoding(1251)?您需要使用Encoding.GetEncoding(850),850是标准cmd-page.850不工作!var source=Encoding.GetEncoding(850);var target=Encoding.UTF8;我得到了tString==“wbadmin 1.0-??????????????-????????-????????-??????????“您的命令行不应该包含恢复而不是恢复吗?没关系。我需要对控制台进行编码。我使用俄语版的Windows Server 2012