我有一个方法,不能同时在多个线程中执行(它会写文件)。我不能使用lock
,因为该方法是async
的。如何避免在另一个线程中调用该方法?程序不应该再次调用它,而应该等待前一个调用完成(也是异步的)。
例如,创建一个新的C#控制台应用程序,并使用以下代码:
using System;
using System.IO;
using System.Threading.Tasks;
namespace ConsoleApplication1 {
internal class Program {
private static void Main () {
CallSlowStuff();
CallSlowStuff();
Console.ReadKey();
}
private static async void CallSlowStuff () {
try {
await DoSlowStuff();
Console.WriteLine("Done!");
}
catch (Exception e) {
Console.WriteLine(e.Message);
}
}
private static async Task DoSlowStuff () {
using (File.Open("_", FileMode.Create, FileAccess.Write, FileShare.None)) {
for (int i = 0; i < 10; i++) {
await Task.Factory.StartNew(Console.WriteLine, i);
await Task.Delay(100);
}
}
}
}
}
在这个例子中,第二次调用 CallSlowStuff
会抛出异常,因为它无法访问已经打开的文件。添加 lock
不是一个选项,因为 lock
和 async
不能混合使用。(Main
方法应被视为不可更改的。在真实应用中,CallSlowStuff
是一个可以在任何地方调用的接口方法。)问题:如何使后续对
CallSlowStuff
的调用等待当前正在运行的调用完成,而不阻塞主线程?是否可以只使用异步/等待和任务(以及 Rx)来完成此操作?
CallSlowStuff
改为“async Task”,那么在Main
方法中会出现编译器警告。 您有任何建议可以完全避免“异步void”吗? 这可能吗? 我一直在想,但没有找到任何解决方案。 - AthariWait()
返回的Task
。如果它是GUI应用程序,则async void
事件处理程序将是正确的解决方案。在ASP.NET MVC中,链不必结束,因为控制器方法可以返回Task
。 - svick