RError.com

RError.com Logo RError.com Logo

RError.com Navigation

  • 主页

Mobile menu

Close
  • 主页
  • 系统&网络
    • 热门问题
    • 最新问题
    • 标签
  • Ubuntu
    • 热门问题
    • 最新问题
    • 标签
  • 帮助
主页 / 问题 / 599952
Accepted
neo
neo
Asked:2020-12-05 23:44:44 +0000 UTC2020-12-05 23:44:44 +0000 UTC 2020-12-05 23:44:44 +0000 UTC

异步读/写到 C# .Net 3.5 进程

  • 772

我在这个过程中实现了异步读/写操作的功能。但是,不会执行提供给进程的命令。

创建一个进程并重定向必要的线程:

StringBuilder outputText = new StringBuilder();
StringBuilder outputError = new StringBuilder();
Process process = new Process();
ProcessStartInfo startInfo = new ProcessStartInfo
{
    FileName = "cmd.exe",
    RedirectStandardInput = true,
    RedirectStandardError = true,
    RedirectStandardOutput = true,
    CreateNoWindow = true,
    UseShellExecute = false,
    ErrorDialog = false,
    Arguments = "/cpython.exe"
};
process.StartInfo = startInfo;

AutoResetEvent outputWaitHandle = new AutoResetEvent(false);
AutoResetEvent errorWaitHandle = new AutoResetEvent(false);

process.OutputDataReceived += (sender, e) => 
{
    if (e.Data == null) { outputWaitHandle.Set();}
    else{ outputText.AppendLine(e.Data); }
}

process.ErrorDataReceived += (sender, e) => 
{
    if (e.Data == null) { errorWaitHandle.Set();}
    else{ outputError.AppendLine(e.Data); }
}

process.Start();
process.BeginOutputReadLine();
process.BeginErrorReadLine();

if (timeout > 0)
{
    if (process.WaitForExit(timeout) &&
        outputWaitHandle(timeout) && 
        errorWaitHandle.WaitOne(timeout))
    {
        // Do some work...
    }
}

异步写入流:

if (process.StandardInput.BaseStream.CanWrite)
{
    byte[] bytesToWrite = Encoding.ASCII.GetBytes("print 'Hello'");
    IAsyncResult handle = process.StandardInput.BaseStream.BeginWrite(bytesToWrite, 0, bytesToWrite.Length, EndWriteCallBack, process.StandardInput.BaseStream);

    if (handle.IsCompleted)
    {
        // Do smth ....
    }
    else
    {
         handle.AsyncWaitHandle.WaitOne();
    }
}

//Callback func
void EndWriteCallBack(IAsyncResult handle)
{
    process.StandardInput.BaseStream.EndWrite(handle);
}

在输出中,我希望我的异步输出将打印的字符串返回给我:

outputText = "print 'Hello'"

就像控制台一样。但是,输出始终为空。告诉我可能是什么问题?

PS:如果你删除输入流的重定向,一切正常。但我希望能够将多个命令发送到同一个控制台,因为这可以通过直接使用它来完成。

c#
  • 1 1 个回答
  • 10 Views

1 个回答

  • Voted
  1. Best Answer
    neo
    2020-12-07T22:07:41Z2020-12-07T22:07:41Z

    在这里找到了我的问题的答案(感谢作者,++++ to karma!):
    Asynchronous I/O to the console, a fully interactive program

    稍微更正了代码(添加了一个实现委托+注释的方法),我得到了这个:

    using System;
    using System.Diagnostics;
    
    public class СmdShell
    {
        private Process shellProcess;
    
        //Делегат
        public delegate void onDataHandler(СmdShell sender, string e);
        //Событие "получение новых данных"
        public event onDataHandler onData;
    
        public СmdShell()
        {
            try
            {
                shellProcess = new Process();
                ProcessStartInfo si = new ProcessStartInfo("cmd.exe");
    
                //Если хотите оставить консоль живой после посланной вами команды
                //Можно ещё передать "/c" - выполнится команда с последующим завершением
                //Либо вообще ничего не передавать
                //Оставлю ссылку после кода на то, какие ещё входные параметры можно задать консоли
                si.Arguments = "/k";
    
                //Перенаправление всех потоков, чтобы можно было асинхронное читать выходные потоки
                //и писать во входной поток
                si.RedirectStandardInput = true;
                si.RedirectStandardOutput = true;
                si.RedirectStandardError = true;
    
                //Обязательно "false", иначе редиректы не будут работать
                si.UseShellExecute = false;
    
                //Чтобы не создавались лишние окошки консоли
                si.CreateNoWindow = true;
    
                //Знать, из какой директории запускаться
                si.WorkingDirectory = Environment.GetEnvironmentVariable("WINDIR");
    
                //Сообщаем процессу стартовую информацию
                shellProcess.StartInfo = si;
    
                //События обработки из потоков вывода
                shellProcess.OutputDataReceived += shellProcess_OutputDataReceived;
                shellProcess.ErrorDataReceived += shellProcess_ErrorDataReceived;
    
                //Стартуем процесс
                shellProcess.Start();
    
                //Начинаем читать выходные потоки
                shellProcess.BeginErrorReadLine();
                shellProcess.BeginOutputReadLine();
            }
            catch (Exception ex)
            {
                Trace.WriteLine(ex.Message);
            }
        }
    
        void shellProcess_ErrorDataReceived(object sender, DataReceivedEventArgs e)
        {
            doOnData(e.Data);
        }
    
        void shellProcess_OutputDataReceived(object sender, DataReceivedEventArgs e)
        {
            doOnData(e.Data);
        }
    
        private void doOnData(string data)
        {
            if (onData != null) onData(this, data);
        }
    
        public void write(string data)
        {
            try
            {
                shellProcess.StandardInput.WriteLine(data);
            }
            catch (Exception ex)
            {
                Trace.WriteLine(ex.Message);
            }
        }
    
        public void test_onData(СmdShell sender, string e)
        {
            Console.WriteLine(e);
        }
    }
    

    测试程序:

    using System;
    public class Program
    {
        static void Main(string[] args)
        {
            CmdShell test = new CmdShell();
            test.onData += test.test_onData;
    
            while (true)
            {
                Console.WriteLine("Enter command:");
                string command = Console.ReadLine();
                test.write(command);
            }
        }
    }
    

    注意,问题!

    将“python.exe”作为命令传递仍然不会返回输出。但是,如果您故意发送无效命令,则将返回 python 返回的错误。

    问题! 你如何获得所有的 Python 输出???

    PS:链接到运行 cmd.exe 的参数:
    CMD - 启动 windows 命令解释器的新副本

    • 0

相关问题

Sidebar

Stats

  • 问题 10021
  • Answers 30001
  • 最佳答案 8000
  • 用户 6900
  • 常问
  • 回答
  • Marko Smith

    如何停止编写糟糕的代码?

    • 3 个回答
  • Marko Smith

    onCreateView 方法重构

    • 1 个回答
  • Marko Smith

    通用还是非通用

    • 2 个回答
  • Marko Smith

    如何访问 jQuery 中的列

    • 1 个回答
  • Marko Smith

    *.tga 文件的组重命名(3620 个)

    • 1 个回答
  • Marko Smith

    内存分配列表C#

    • 1 个回答
  • Marko Smith

    常规赛适度贪婪

    • 1 个回答
  • Marko Smith

    如何制作自己的自动完成/自动更正?

    • 1 个回答
  • Marko Smith

    选择斐波那契数列

    • 2 个回答
  • Marko Smith

    所有 API 版本中的通用权限代码

    • 2 个回答
  • Martin Hope
    jfs *(星号)和 ** 双星号在 Python 中是什么意思? 2020-11-23 05:07:40 +0000 UTC
  • Martin Hope
    hwak 哪个孩子调用了父母的静态方法?还是不可能完成的任务? 2020-11-18 16:30:55 +0000 UTC
  • Martin Hope
    Qwertiy 并变成3个无穷大 2020-11-06 07:15:57 +0000 UTC
  • Martin Hope
    koks_rs 什么是样板代码? 2020-10-27 15:43:19 +0000 UTC
  • Martin Hope
    user207618 Codegolf——组合选择算法的实现 2020-10-23 18:46:29 +0000 UTC
  • Martin Hope
    Sirop4ik 向 git 提交发布的正确方法是什么? 2020-10-05 00:02:00 +0000 UTC
  • Martin Hope
    Arch ArrayList 与 LinkedList 的区别? 2020-09-20 02:42:49 +0000 UTC
  • Martin Hope
    iluxa1810 哪个更正确使用:if () 或 try-catch? 2020-08-23 18:56:13 +0000 UTC
  • Martin Hope
    faoxis 为什么在这么多示例中函数都称为 foo? 2020-08-15 04:42:49 +0000 UTC
  • Martin Hope
    Pavel Mayorov 如何从事件或回调函数中返回值?或者至少等他们完成。 2020-08-11 16:49:28 +0000 UTC

热门标签

javascript python java php c# c++ html android jquery mysql

Explore

  • 主页
  • 问题
    • 热门问题
    • 最新问题
  • 标签
  • 帮助

Footer

RError.com

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

帮助

© 2023 RError.com All Rights Reserve   沪ICP备12040472号-5