Async.Catch 不适用于 OperationCanceledException

2023-11-23

我使用 Async.Catch 来处理异步工作流程引发的异常:

work
|> Async.Catch
|> Async.RunSynchronously
|> fun x -> match x with
            | Choice1Of2 _ -> () // success
            | Choice2Of2 ex -> // failure, handle exception

今天我注意到 Async.Catch 不处理 OperationCanceledExceptions。异常没有从 Async.Catch 中获取 Choice,而是不断冒泡,直到它击中我为止。我预计以下测试是红色的,但它是绿色的:

  [<Test>]
  let ``Async.Catch doesnt work on OperationCancelledExceptions``() =
    use cancellationTokenSource = new System.Threading.CancellationTokenSource(1000)

    let work = async {
      while true do
        do! Async.Sleep 100
    }

    (fun () -> work
               |> Async.Catch
               |> fun x -> Async.RunSynchronously (x, cancellationToken=cancellationTokenSource.Token)
               |> ignore)
    |> should throw typeof<System.OperationCanceledException>

使用 Async.Catch + Choices + 匹配评估一些异常,以及使用 try/catch 块评估其他一些异常似乎并不正确......它看起来像下面这样,这太复杂了。除此之外,我想知道 Async.Catch 有什么用处,因为无论如何我都必须使用 try/catch 块......:

  [<Test>]
  let ``evaluating exceptions of async workflows``() =
    use cancellationTokenSource = new System.Threading.CancellationTokenSource(1000)

    let work = async {
      while true do
        do! Async.Sleep 100
    }

    try
      work
      |> Async.Catch
      |> fun x -> Async.RunSynchronously (x, cancellationToken=cancellationTokenSource.Token)
      |> fun x -> match x with
                  | Choice1Of2 result -> () // success, process result
                  | Choice2Of2 ex -> () // failure, handle exception
    with ex -> () // another failure, handle exception here too

处理异步工作流程异常的最佳方法是什么?我应该转储 Async.Catch 并在各处使用 try/catch 块吗?


取消是异步计算中的一种特殊异常。当工作流被取消时,这也会取消所有子计算(取消令牌是共享的)。因此,如果您可以将取消作为普通异常处理,它仍然可以取消计算的某些其他部分(并且很难推断正在发生的情况)。

但是,您可以编写一个原语starts一个工作流程(并将其与父工作流程分开),然后在此子工作流程中处理取消。

type Async = 
  static member StartCatchCancellation(work, ?cancellationToken) = 
    Async.FromContinuations(fun (cont, econt, _) ->
      // When the child is cancelled, report OperationCancelled
      // as an ordinary exception to "error continuation" rather
      // than using "cancellation continuation"
      let ccont e = econt e
      // Start the workflow using a provided cancellation token
      Async.StartWithContinuations( work, cont, econt, ccont, 
                                    ?cancellationToken=cancellationToken) )

用法类似于Async.Catch,但您必须将取消令牌传递给StartCatchCancellation而不是将其传递给主函数RunSynchronously(因为工作流程是单独启动的):

let work = 
  async { while true do
            do! Async.Sleep 100 }

let ct = new System.Threading.CancellationTokenSource(10000)
Async.StartCatchCancellation(work, ct.Token) 
|> Async.Catch
|> Async.RunSynchronously 
|> printfn "%A"
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Async.Catch 不适用于 OperationCanceledException 的相关文章

随机推荐