【发布时间】:2020-10-19 22:11:49
【问题描述】:
我有一个不能同时在多个线程中执行的方法(它写入文件)。我不能使用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 进行后续调用等待当前正在运行的调用完成?可以只使用 async/await 和任务(也许还有 Rx)来完成吗?
【问题讨论】:
-
顺便说一句,你真的要让你的方法
async void吗?这通常是个坏主意,除非它是和事件处理程序。 -
@svick "Async" 方法链必须在某处停止。如果我创建
CallSlowStuffasync Task,那么我将在Main方法中收到编译器警告。您对如何完全避免async void有什么建议吗?甚至可能吗?我对此感到疑惑,但没有找到任何解决方案。 -
这取决于您的应用程序的类型。如果你真的有一个控制台应用程序,正确的解决方案是为返回的
Task同步Wait()。如果它是一个 GUI 应用程序,那么async void事件处理程序将是正确的解决方案。在 ASP.NET MVC 中,链不必结束,因为控制器方法可以返回Task。
标签: c# .net locking task-parallel-library async-await