0

0

c# 如何使用 Profiler 工具分析c#并发瓶颈

幻夢星雲

幻夢星雲

发布时间:2026-01-25 09:03:10

|

119人浏览过

|

来源于php中文网

原创

Visual Studio CPU 使用率工具可定位并发争用:勾选收集争用数据,按 Blocked Time 排序,关注 Monitor.Enter 等锁调用栈;dotnet-trace + PerfView 分析线程池队列堆积;dotnet-dump 查锁持有链;避免将异步等待误判为瓶颈。

c# 如何使用 profiler 工具分析c#并发瓶颈

用 Visual Studio 的 CPU 使用率工具定位并发线程争用

Visual Studio 自带的 CPU 使用率 工具(非第三方 Profiler)是排查 C# 并发瓶颈最直接的方式。它能显示线程状态(Running / Blocked / Waiting)、调用堆、以及哪些方法在锁上卡住。关键不是看“谁耗 CPU 高”,而是找大量线程长期处于 WaitBlocked 状态的热点

  • 启动时勾选 Collect .NET allocation and contention data(否则看不到锁等待和线程阻塞详情)
  • 重现高并发场景后,切换到 Threads 视图,按 Blocked Time (ms) 降序排列,重点关注 Monitor.EnterConcurrentQueue.EnqueueSpinWait.SpinOnce 等调用栈
  • 若看到多个线程在同一个 object 实例上调用 Monitor.Enter,说明存在锁争用;若堆栈指向 lock (_syncRoot)Monitor.Enter(_locker),且 _locker 是共享实例,就是瓶颈点

用 dotnet-trace + PerfView 分析 ThreadPool 饱和与队列堆积

并发请求激增但吞吐不升反降,常因线程池任务排队过长,而非 CPU 不够。此时 dotnet-trace 可捕获 Microsoft-Extensions-LoggingMicrosoft-DotNet-ILCompiler 之外的关键 provider:Microsoft-DotNet-ThreadPool

  • 命令行采集:
    dotnet-trace collect --providers Microsoft-DotNet-ThreadPool:0x1000000000000000:4,Microsoft-DotNet-ThreadPool:0x2000000000000000:4 --process-id 
  • 在 PerfView 中打开 ThreadPool > WorkerThreadQueueLength 计数器,观察是否持续 > 0;若平均值 > 5,说明任务提交速率远超线程处理能力
  • 检查 ThreadPool > WorkerThreadStart 事件间隔:若新线程启动延迟 > 100ms,说明系统已接近线程池上限(默认 max worker threads = min(1000, #cores × 500)),需调用 ThreadPool.SetMaxThreads 或改用 Task.Run + 自定义 TaskScheduler

dotnet-dump 查看运行时线程锁持有链

当应用挂起或响应极慢,且怀疑死锁或长持锁,dotnet-dump 比实时 Profiler 更可靠——它能抓取完整托管堆和线程上下文,包括每个线程当前持有的 Monitor 和正在等待的 obj 地址。

图星人
图星人

好用的AI生图工具,百万免费商用图库

下载
  • 生成 dump:
    dotnet-dump collect -p 
  • 分析锁关系:
    dotnet-dump analyze  -c "threads -s" | findstr "Lock"
    找出状态为 Waiting on lock 的线程
  • 对任一线程 ID(如 0x1a),执行:
    dumpheap -stat
    +
    dumpobj 
    确认该对象是否被其他线程 Monitor.Enter 后未释放
  • 特别注意 ConcurrentDictionary 的分段锁:若多个线程反复操作同一 key(尤其 hash 冲突高时),可能挤在同一个 segment 上,表现为单个 object 被频繁 Monitor.Enter

避免误判:别把异步 I/O 等待当成并发瓶颈

await Task.Delayawait httpClient.GetAsync 或 EF Core 的 ToListAsync 等操作,在 Profiler 中常显示为线程“空闲”或“未运行”,但这不是瓶颈——它们本就不该占 CPU。真正要盯的是那些本该异步却用了同步阻塞的调用。

  • 检查是否误用 .Result.Wait():Profiler 中会显示线程在 Task.InternalWaitThreadPool.WaitCallback 上长时间 Blocked
  • 确认数据库连接是否复用:若每个请求都新建 SqlConnection 且未及时 Dispose,连接池耗尽会导致后续请求在 SqlConnection.Open 上无限等待(表现类似锁争用,但根源是资源池)
  • 警惕 ValueTask 误用:重复 await 同一个已完成的 ValueTask 会抛 InvalidOperationException,某些 Profiler 会将其归类为“异常开销”,掩盖真实问题

并发瓶颈往往不在代码写法多炫酷,而在共享状态的粒度是否合理、线程资源是否被隐式耗尽、以及同步/异步边界是否被无意打破。跑一次 dotnet-trace + dotnet-dump 组合,比读十篇锁优化文章更能定位你进程里那个卡住的 object 实例。

相关专题

更多
堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

394

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

574

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

394

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

574

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

482

2023.08.10

数据库三范式
数据库三范式

数据库三范式是一种设计规范,用于规范化关系型数据库中的数据结构,它通过消除冗余数据、提高数据库性能和数据一致性,提供了一种有效的数据库设计方法。本专题提供数据库三范式相关的文章、下载和课程。

356

2023.06.29

如何删除数据库
如何删除数据库

删除数据库是指在MySQL中完全移除一个数据库及其所包含的所有数据和结构,作用包括:1、释放存储空间;2、确保数据的安全性;3、提高数据库的整体性能,加速查询和操作的执行速度。尽管删除数据库具有一些好处,但在执行任何删除操作之前,务必谨慎操作,并备份重要的数据。删除数据库将永久性地删除所有相关数据和结构,无法回滚。

2076

2023.08.14

vb怎么连接数据库
vb怎么连接数据库

在VB中,连接数据库通常使用ADO(ActiveX 数据对象)或 DAO(Data Access Objects)这两个技术来实现:1、引入ADO库;2、创建ADO连接对象;3、配置连接字符串;4、打开连接;5、执行SQL语句;6、处理查询结果;7、关闭连接即可。

348

2023.08.31

c++ 根号
c++ 根号

本专题整合了c++根号相关教程,阅读专题下面的文章了解更多详细内容。

25

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 4.1万人学习

Pandas 教程
Pandas 教程

共15课时 | 1.0万人学习

ASP 教程
ASP 教程

共34课时 | 4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号