Как я могу перенаправить stdout IronPython в С#?

У меня есть следующее:

public partial class Form1 : Form
{
    public Form1()
    {
        InitializeComponent();
    }

    private void button3_Click(object sender, EventArgs e)
    {
        try
        {
            var strExpression = @"
            import sys
            sys.stdout=my.write
            print 'ABC'
            ";
            var engine = Python.CreateEngine();
            var scope = engine.CreateScope();
            var sourceCode = engine.CreateScriptSourceFromString(strExpression);
            scope.SetVariable("my", this);
            var actual = sourceCode.Execute<string>(scope);
            textBox1.Text += actual;
        }
        catch (System.Exception ex)
        {
            MessageBox.Show(ex.ToString());
        }
    }

    public void write(string s)
    {
        textBox1.Text += s;
    }
}

Но я получаю Exception, который говорит, что нет write.

Что я делаю неправильно?

Ответы

Ответ 1

Вы можете установить поток и текстовый редактор непосредственно из С#:

engine.Runtime.IO.SetOutput(stream, txtWriter);
engine.Runtime.IO.SetErrorOutput(stream, txtWriter);

Чтобы перенаправить вывод, например, вы можете переопределить класс TextWriter с новой записью в текстовом поле.


например.

в моем приложении я переопределил класс StreamWriter, который увеличивает события, когда что-то написано в потоке (здесь только часть кода):

public class MyEvtArgs<T> : EventArgs
{
    public T Value
    {
        get;
        private set;
    }
    public MyEvtArgs(T value)
    {
        this.Value = value;
    }
}


public class EventRaisingStreamWriter : StreamWriter
{
    #region Event
    public event EventHandler<MyEvtArgs<string>> StringWritten;
    #endregion

    #region CTOR
    public EventRaisingStreamWriter(Stream s):base(s)
    { }
    #endregion

    #region Private Methods
    private void LaunchEvent(string txtWritten)
    {
        if (StringWritten != null)
        {
            StringWritten(this, new MyEvtArgs<string>(txtWritten));
        }
    }
    #endregion


    #region Overrides

    public override void Write(string value)
    {
        base.Write(value);
        LaunchEvent(value);
    }
    public override void Write(bool value)
    {
        base.Write(value);
        LaunchEvent(value.ToString());
    }
    // here override all writing methods...

    #endregion
}

Затем в вашем приложении вы должны просто сделать что-то вроде:

    MemoryStream ms = new MemoryStream();

    outputWr = new EventRaisingStreamWriter(ms);
    outputWr.StringWritten += new EventHandler<MyEvtArgs<string>>(sWr_StringWritten);

    engine.Runtime.IO.SetOutput(ms, errorWr);



    void sWr_StringWritten(object sender, MyEvtArgs<string> e)
    {
        textBox1.Text += s.Value;
    }

Ответ 2

Ваш пример близок к работе.

Проблема, которую вы видели, заключается в том, что sys.stdout=my.write должен быть sys.stdout=my. Также представляется, что Python ожидает найти логический атрибут softspace.

Я сделал эти два изменения в коде ниже. Надеюсь, теперь это должно работать так, как вы ожидали.

public partial class Form1 : Form
{
    public Form1()
    {
        InitializeComponent();
    }

    private void button3_Click(object sender, EventArgs e)
    {
        try
        {
            var strExpression = @"
import sys
sys.stdout=my
print 'ABC' ";

            var engine = Python.CreateEngine();
            var scope = engine.CreateScope();
            var sourceCode = engine.CreateScriptSourceFromString(strExpression);
            scope.SetVariable("my", this);
            var actual = sourceCode.Execute(scope);
            textBox1.Text += actual;
        } catch (System.Exception ex) {
            MessageBox.Show(ex.ToString());
        }
    }

    public bool softspace;

    public void write(string s)
    {
        textBox1.Text += s;
    }
}

Ответ 3

Это сработало отлично для меня

        pyRuntime = Python.CreateRuntime();
        pyRuntime.IO.SetOutput(Console.OpenStandardOutput(), new UTF8Encoding(true, false));