此表达式应具有 Async<'a> 类型,但此处具有类型 DispatcherOperation

Nav*_*rma 5 f# async-await

我正在构建一个包含 F# 文件的项目,在构建过程中,第 39 行出现错误:

此表达式应具有 Async<'a> 类型,但此处具有类型 DispatcherOperation

open System
open System.Collections.Generic
open System.Collections.ObjectModel
open System.Linq
open System.Net
open System.Reactive.Disposables
open System.Runtime.CompilerServices
open System.Threading
open System.Windows
open System.Windows.Threading

open Microsoft.Practices.Unity

type IActivityContext<'TResult> = interface
    abstract container: IUnityContainer
    abstract Success: result:'TResult -> unit
    abstract Error: error:Exception -> unit
    abstract Cancel: unit -> unit
    abstract RegisterCancellationCallback: callback:Action->IDisposable
end

[<Extension>]
type ActivityExtensions private() = class
    [<Extension>]
    static member StartViewActivity<'TResult>(container: IUnityContainer, callback: Action<IActivityContext<'TResult>>)= async{
        let! ct = Async.CancellationToken
        return! Async.FromContinuations(fun (success, error, cancel) ->
            let context = {
                new IActivityContext<'TResult> with
                    member this.container = container
                    member this.Success(result:'TResult) = success(result)
                    member this.Error(err:Exception) = error(err)
                    member this.Cancel() = cancel(new OperationCanceledException())
                    member this.RegisterCancellationCallback(callback: Action) = 
                        ct.Register(callback) :> IDisposable
            }
            let disp = Application.Current.Dispatcher
            Async.StartWithContinuations(
(* ERROR -> *)  disp.InvokeAsync((fun() -> callback.Invoke(context))),
                (fun()->()), 
                error, 
                cancel,
                ct
            )
        )
    }
end
Run Code Online (Sandbox Code Playgroud)

有谁知道为什么会出现此错误以及解决方案是什么?

Car*_*Dev 5

错误消息说明了一切:Dispatcher.InvokeAsync返回 aDispatcherOperationAsync.StartWithContinuations期望Async.

您可以等待DispatcherOperation.Task

Async.StartWithContinuations(
    disp.InvokeAsync((fun() -> callback.Invoke(context))).Task |> Async.AwaitTask,
    ignore,
    error,
    cancel,
    ct)
Run Code Online (Sandbox Code Playgroud)

请注意,用 with 包装表达式async不会自动使其异步并且包装一个已经异步的流而不等待两次甚至可能根本不会执行内部异步:

let print = async { printfn "Can we observe this?" }

async { return print } |> Async.RunSynchronously // nothing printed
async { return! print } |> Async.RunSynchronously // prints message
Run Code Online (Sandbox Code Playgroud)

所以这不是异步的:

Async.StartWithContinuations(
    async { disp.Invoke((fun() -> callback.Invoke(context))) },
    ignore,
    error,
    cancel,
    ct)
Run Code Online (Sandbox Code Playgroud)

这甚至可能不会调用回调(取决于InvokeAsync实现,但Tasks 通常是隐式启动的):

Async.StartWithContinuations(
    async { disp.InvokeAsync((fun() -> callback.Invoke(context))) },
    ignore, 
    error, 
    cancel,
    ct)
Run Code Online (Sandbox Code Playgroud)

最重要的是,ignore可能会内部回调返回之前调用回调(通常是成功的回调,这里是 : )。