gpt4 book ai didi

c# - Nito.AsyncEx.AsyncLock 堆栈溢出,有很多等待者和同步快速路径

转载 作者:太空宇宙 更新时间:2023-11-03 21:08:12 25 4
gpt4 key购买 nike

我正在使用 AsyncLock来自 Stephen Cleary 的 Nito.AsyncEx NuGet package (v3.0.1) 保护昂贵资源的初始化,因此只有第一个调用者会执行耗时的异步初始化,所有后续调用者将异步等待直到初始化完成,然后获取缓存资源。

我首先注意到的是受 AsyncLock 保护的区域之后的代码正在以任务开始的精确相反顺序执行任务(即最后一个任务开始必须首先继续通过锁定区域,然后是倒数第二个任务,依此类推直到第一个任务最后继续)。

然后在调查为什么会发生这种情况的过程中,我发现当有大量异步任务时,我总是会出现堆栈溢出。这是一个简化的示例:

object _foo;
readonly Nito.AsyncEx.AsyncLock _fooLock = new Nito.AsyncEx.AsyncLock();

async Task<object> GetFooAsync()
{
using (await _fooLock.LockAsync().ConfigureAwait(false))
{
if (_foo == null)
{
// Simulate time-consuming asynchronous initialization,
// during which all the subsequent tasks end up awaiting the AsyncLock.
await Task.Delay(5000).ConfigureAwait(false);
_foo = new object();
}
return _foo;
}
}

async Task DoStuffAsync()
{
object foo = await GetFooAsync().ConfigureAwait(false);
// Do stuff with foo...
}

void DoStuff()
{
var tasks = new List<Task>();

for (int i = 1; i <= 1000; i++)
{
tasks.Add(DoStuffAsync());
}

Task.WhenAll(tasks).Wait();
}

如果快速路径通过GetFooAsync()不是同步的(例如,如果我在 await Task.Yield(); 之前添加 return _foo;),那么不仅不会发生堆栈溢出,而且任务会按照它们开始的顺序继续经过锁定区域。

我可能会更改我的代码以使用 AsyncLazy<T>来自 AsyncEx 而不是这个用例,我已经测试过并且似乎没有出现这个问题。

但是,我想知道这个问题是否是由于我的代码错误造成的,AsyncLock 中的错误,还是只是预期的行为(更多的陷阱)?

最佳答案

这是一个bug in AsyncLock ;所有基于队列的异步协调原语都有同样的问题。修复工作正在进行中。

new version of this library有一个重写的队列,不会遇到这个问题。

关于c# - Nito.AsyncEx.AsyncLock 堆栈溢出,有很多等待者和同步快速路径,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39779088/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com