实用技巧

关注公众号 jb51net

关闭
首页 > 网络编程 > ASP.NET > 实用技巧 > .NET .Result 死锁与线程池饥饿

.NET .Result 不同框架下的死锁与线程池饥饿问题解析

作者:ryan-deng

这篇文章主要介绍了.NET .Result 不同框架下的死锁与线程池饥饿问题解析,本文给大家介绍的非常详细对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧

这篇只讲一个知识点:在 .NET 代码里用 .Result(或 GetAwaiter().GetResult())同步阻塞异步任务,为什么在不同框架下会触发不同类型的事故。

问题背景

同样一行代码,在两个系统里出现了完全不同的故障:

两边都有这段写法:

public string GetData()
{
return GetDataAsync().Result;
}
private async Task<string> GetDataAsync()
{
await Task.Delay(50);
return "ok";
}

原理:同一个坑,两种后果

场景 1:ASP.NET Classic / WinForms / WPF(有 SynchronizationContext)

这类框架默认要求 continuation 回到原上下文(UI 线程或请求上下文)。

.Result 先把当前线程阻塞住,Task 完成后 continuation 又想回到这条线程,结果互相等待:

所以你会看到"请求一直转圈"或"界面完全卡死"。

场景 2:ASP.NET Core(默认无 SynchronizationContext)

在默认配置下,ASP.NET Core 没有传统的请求级 SynchronizationContext,所以通常不会触发上面的经典互锁。

它会把线程池工作线程同步阻塞住。并发一上来,越来越多线程被卡在 .Result,线程池来不及补充,新请求拿不到线程,就出现线程饥饿:

这就是"看起来不像死锁,但系统几乎不可用"的典型表现。

最小对照示例

public sealed class DemoService
{
// ❌ 错误:同步包装异步
public int GetNumber()
{
return GetNumberAsync().Result;
}
// ✅ 正确:异步到底
public async Task<int> GetNumberAsync()
{
await Task.Delay(10);
return 42;
}
}

如何避坑(只保留最关键三条)

一句结论

.Result 在老框架里更容易直接死锁,在 ASP.NET Core 里更容易演化成线程池饥饿;表现不同,本质相同,都是"阻塞等待异步"导致的。

到此这篇关于.NET .Result 不同框架下的死锁与线程池饥饿问题解析的文章就介绍到这了,更多相关.NET .Result 死锁与线程池饥饿内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文