gpt4 book ai didi

c# - 如何从 C# 调用和处理异步 F# 工作流

转载 作者:行者123 更新时间:2023-11-30 19:48:17 25 4
gpt4 key购买 nike

我读过一些 F# 教程,我注意到与 C# 相比,在 F# 中执行异步和并行编程有多么容易。因此,我正在尝试编写一个 F# 库,它将从 C# 调用并将 C# 函数(委托(delegate))作为参数并异步运行它。

到目前为止,我已经设法传递了函数(我什至可以取消),但我想念的是如何实现回调到 C#,一旦异步操作完成,回调就会被执行。 (例如函数 AsynchronousTaskCompleted?)。另外我想知道我是否可以从 AsynchronousTask 函数将(例如 Progress %)发回 F#。

有人可以帮帮我吗?

这是我到目前为止编写的代码(我不熟悉 F#,所以以下代码可能是错误的或实现不佳)。

//C# Code Implementation (How I make the calls/handling)
//Action definition is: public delegate void Action();
Action action = new Action(AsynchronousTask);
Action cancelAction = new Action(AsynchronousTaskCancelled);
myAsyncUtility.StartTask2(action, cancelAction);
Debug.WriteLine("0. The task is in progress and current thread is not blocked");
......
private void AsynchronousTask()
{
//Perform a time-consuming task
Debug.WriteLine("1. Asynchronous task has started.");
System.Threading.Thread.Sleep(7000);
//Post progress back to F# progress window?
System.Threading.Thread.Sleep(2000);
}
private void AsynchronousTaskCompleted(IAsyncResult asyncResult)
{
Debug.WriteLine("2. The Asynchronous task has been completed - Event Raised");
}
private void AsynchronousTaskCancelled()
{
Debug.WriteLine("3. The Asynchronous task has been cancelled - Event Raised");
}

//F# Code Implementation
member x.StartTask2(action:Action, cancelAction:Action) =
async {
do! Async.FromBeginEnd(action.BeginInvoke, action.EndInvoke, cancelAction.Invoke)
}|> Async.StartImmediate
do printfn "This code should run before the asynchronous operation is completed"
let progressWindow = new TaskProgressWindow()
progressWindow.Run() //This class(type in F#) shows a dialog with a cancel button
//When the cancel button is pressed I call Async.CancelDefaultToken()

member x.Cancel() =
Async.CancelDefaultToken()

最佳答案

要获得 F# 异步工作流的好处,您必须实际用 F# 编写异步计算。您尝试编写的代码将无法运行(即它可能会运行,但不会有用)。

在 F# 中编写异步计算时,可以使用 let! 进行异步调用和 do! .这允许您使用其他原始的非阻塞计算。例如,您可以使用 Async.Sleep而不是 Thread.Sleep .

// This is a synchronous call that will block thread for 1 sec
async { do Thread.Sleep(1000)
someMoreStuff() }

// This is an asynchronous call that will not block threads - it will create
// a timer and when the timer elapses, it will call 'someMoreStuff'
async { do! Async.Sleep(1000)
someMoreStuff() }

您只能在 async 中使用异步操作 block ,它依赖于 F# 编译器处理 do! 的方式和 let! .对于以顺序方式编写的代码(例如,在 C# 中或在 F# 中的 async block 之外),没有(简单的)方法来获得真正的非阻塞执行。

如果您想使用 F# 获得异步工作流的好处,那么最好的选择是在 F# 中实现操作,然后使用 Async.StartAsTask 将它们公开给 C# (它为您提供 C# 可以轻松使用的 Task<T>)。像这样:

let someFunction(n) = async {
do! Async.Sleep(n)
Console.WriteLine("working")
do! Async.Sleep(n)
Console.WriteLine("done")
return 10 }

type AsyncStuff() =
member x.Foo(n) = someFunction(n) |> Async.StartAsTask

// In C#, you can write:
var as = new AsyncStuff()
as.Foo(1000).ContinueWith(op =>
// 'Value' will throw if there was an exception
Console.WriteLine(op.Value))

如果您不想使用 F#(至少对于异步计算的实现而言),那么异步工作流将无济于事。您可以使用 Task 实现类似的事情, BackgroundWorker或其他 C# 技术(但您将失去在不阻塞线程的情况下轻松运行操作的能力)。

关于c# - 如何从 C# 调用和处理异步 F# 工作流,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5677366/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com