我正在尝试重定向控制台应用程序的stdin和stdout,以便我可以通过F#与它们进行交互.但是,根据控制台应用程序,明显的代码似乎失败了.以下F#代码适用dir
但失败(挂起)python
和fsi
:
open System
open System.Diagnostics
let f = new Process()
f.StartInfo.FileName <- "python"
f.StartInfo.UseShellExecute <- false
f.StartInfo.RedirectStandardError <- true
f.StartInfo.RedirectStandardInput <- true
f.StartInfo.RedirectStandardOutput <- true
f.EnableRaisingEvents <- true
f.StartInfo.CreateNoWindow <- true
f.Start()
let line = f.StandardOutput.ReadLine()
Run Code Online (Sandbox Code Playgroud)
这挂起python,但适用于dir.
这是否与使用readline的python和fsi有关,还是我犯了一个明显的错误?有没有可以让我与F#的fsi或python REPL进行交互?
这就是您正在寻找的代码(我在第 9 章“脚本编写”中方便地编写了该代码;)如前所述,ReadLine 会阻塞,直到出现一整行,这会导致各种挂起。最好的选择是挂钩 OutputDataRecieved 事件。
open System.Text
open System.Diagnostics
let shellEx program args =
let startInfo = new ProcessStartInfo()
startInfo.FileName <- program
startInfo.Arguments <- args
startInfo.UseShellExecute <- false
startInfo.RedirectStandardOutput <- true
startInfo.RedirectStandardInput <- true
let proc = new Process()
proc.EnableRaisingEvents <- true
let driverOutput = new StringBuilder()
proc.OutputDataReceived.AddHandler(
DataReceivedEventHandler(
(fun sender args -> driverOutput.Append(args.Data) |> ignore)
)
)
proc.StartInfo <- startInfo
proc.Start() |> ignore
proc.BeginOutputReadLine()
// Now we can write to the program
proc.StandardInput.WriteLine("let x = 1;;")
proc.StandardInput.WriteLine("x + x + x;;")
proc.StandardInput.WriteLine("#q;;")
proc.WaitForExit()
(proc.ExitCode, driverOutput.ToString())
Run Code Online (Sandbox Code Playgroud)
输出(可以进行美化):
val it : int * string =
(0,
"Microsoft F# Interactive, (c) Microsoft Corporation, All Rights ReservedF# Version 1.9.7.8, compiling for .NET Framework Version v2.0.50727For help type #help;;> val x : int = 1> val it : int = 3> ")
Run Code Online (Sandbox Code Playgroud)