首页 > 解决方案 > Async Sub() 或 Async Function() 作为 Fire and Forget 的任务?

问题描述

我想异步做一些事情,不关心结果。
最好的方法是什么?

Public Async Function HandlerForSomeEvent() As Task
  'This is where I do stuff

  'Then this is stuff I want to do without waiting for it
  DoStuff()

  'Here I continue doing other stuff
End Function
Async Sub DoStuff()
  'Doing stuff while the handler continues doing it's stuff
End Sub

'VS

Async Function DoStuff() As Task
  'Doing stuff while the handler continues doing it's stuff
End Function

每个人都告诉我使用Function As Task,但由于我不等待它,我总是在 VS 中收到烦人的警告。
有什么区别,我为什么要这样做?

标签: vb.netasynchronousasync-awaittask

解决方案


Function使用 a返回 aTask而不是a 的主要原因Async Sub有两个。首先是异常处理。如果你做这样的事情,你会得到一个未捕获的异常,这将终止你的应用程序:

Public Sub Main()
    Try
        DoSomethingAsync()
    Catch ex As Exception
        Console.WriteLine(ex)
    End Try
End Sub

Private Async Sub DoSomethingAsync()
    Throw New Exception()
End Sub

Try/Catch 块不会捕获异常,因为它是在单独的线程上引发的。

但是,如果您这样做,它将被捕获并处理:

Public Async Sub Main()
    Try
        Await DoSomethingAsync()
    Catch ex As Exception
        Console.WriteLine(ex)
    End Try
End Sub

Private Async Function DoSomethingAsync() As Task
    Throw New Exception()
End Sub

第二个重要原因是,即使您现在可能不需要它,但以后可能会需要它。如果它已经是一个返回任务的函数,那么您可以随意使用它。如果你不这样做,你就不能等待它,或者安排在它之后发生的事情,或者中止它,或者做任何其他你可以用任务做的事情。所以,问题真的不是“为什么要这样做?”,而是“为什么不这样做?”

例如,这会输出“Here”,然后是“Done”:

Public Sub Main()
    DoSomethingAsync()
    Threading.Thread.Sleep(5000)
    Console.WriteLine("Done")
End Sub

Public Async Function DoSomethingAsync() As Task
    Console.WriteLine("Here")
End Function

正如其他人所提到的,为了避免警告,您可以将任务分配给这样的变量:

Public Sub Main()
    Dim t As Task = DoSomethingAsync()
    Threading.Thread.Sleep(5000)
    Console.WriteLine("Done")
End Sub

无论哪种方式,它的工作方式都相同,并在自己的线程上运行,就像Async Sub. Task变量是否超出范围也没关系。即使没有任何东西引用它,该任务也会继续运行。例如:

Public Sub Main()
    DoSomething()
    Threading.Thread.Sleep(5000)
    Console.WriteLine("Really Done")
End Sub

Public Sub DoSomething()
    Dim t As Task = DoSomethingAsync()
    Console.WriteLine("Done Doing Something")
End Sub

Public Async Function DoSomethingAsync() As Task
    Console.WriteLine("Here")
End Function

推荐阅读