0

0

c# ETW (Event Tracing for Windows) 和 .NET 事件探查并发问题

畫卷琴夢

畫卷琴夢

发布时间:2026-01-14 12:07:02

|

203人浏览过

|

来源于php中文网

原创

ETW采集.NET并发事件时看不到ThreadPoolWorkerThreadStart等,因需显式启用0x800(ThreadPool)关键字;dotnet-dump与ETW交叉验证依赖时间戳对齐和线程ID关联;EventSource丢事件是因默认采样限流;ThreadPoolWorkerThreadStart的ManagedThreadId恒为0,需通过OSThreadId关联Thread/Start事件获取。

c# etw (event tracing for windows) 和 .net 事件探查并发问题

ETW 采集 .NET 并发事件时,为什么看不到 ThreadPoolWorkerThreadStartThreadPoolEnqueue

因为这些事件默认被禁用——.NET 运行时(CoreCLR / .NET 5+)的 ETW provider(Microsoft-Windows-DotNETRuntime)需显式启用「ThreadPool」关键字(keyword),否则即使开启 EventSource 级别,线程池相关事件也不会发出。

实操建议:

  • 使用 dotnet-trace 时加 --providers Microsoft-Windows-DotNETRuntime:0x0000000000000800(十六进制 0x800 对应 ThreadPool 关键字)
  • logman 启动 ETW session 时,provider 配置中必须包含 keywords=0x800,例如:
    logman start mytrace -p "Microsoft-Windows-DotNETRuntime" "0x800" -o trace.etl -ets
  • 在 C# 中用 EventListener 订阅时,重写 OnEventSourceCreated 并对 eventSource.Name == "Microsoft-Windows-DotNETRuntime" 调用 EnableEvents(eventSource, EventLevel.Verbose, (EventKeywords)0x800)

dotnet-dump 和 ETW 日志怎么交叉验证锁竞争?

ETW 提供时间线上的高密度事件(如 MonitorEnterStart/MonitorEnterStopThreadPoolWorkerThreadStart),但不记录托管堆对象地址;而 dotnet-dump 只能捕获某一时刻的快照。二者需靠「时间戳对齐 + 线程 ID 关联」来定位。

关键操作点:

  • 采集 ETW 时务必启用 EventSourceTimeStamp 字段(默认开启),并用 perfviewTraceEvent 库解析出纳秒级时间戳
  • 触发 dump 前,先在代码中插入 System.Diagnostics.Debug.WriteLine($"DUMP_POINT: {DateTime.UtcNow:O} Thread={Thread.CurrentThread.ManagedThreadId}");,让日志与 dump 时间锚定
  • dotnet-dump analyze 查看 clrstack -all,比对线程 ID 和 ETW 中 ManagedThreadId 字段(注意:ETW 事件里的 ThreadId 是 OS 线程 ID,需通过 !threadsdumpheap -stat 中的线程对象反查对应关系)

为什么 EventSource 自定义事件在并发压测下丢失严重?

不是丢,是被限流了。.NET 的 EventSource 默认启用「采样丢弃(sampling discard)」机制:当事件速率超过阈值(约 10k/s),后续事件会被静默丢弃,且不报错。

萝卜简历
萝卜简历

免费在线AI简历制作工具,帮助求职者轻松完成简历制作。

下载

缓解方式:

  • 构造 EventSource 时传入 EventSourceSettings.EtwSelfDescribingEventFormat 以外的选项(如 EventSourceSettings.None),但这会禁用 ETW 自描述格式,需手动维护 manifest
  • 改用异步缓冲模式:在 WriteEvent 前先写入 ConcurrentQueue,再由独立线程批量调用 WriteEventCore,降低单次调用开销
  • 生产环境慎用 EventLevel.LogAlways,优先用 EventLevel.Informational + 条件过滤(例如只在 Monitor.IsEntered(obj) 为 true 时才记录争用)

TraceEvent 库解析 ETW 时,ThreadPoolWorkerThreadStartManagedThreadId 字段总是 0?

这是 .NET Runtime provider 的已知行为:该事件在 CoreCLR 中不填充 ManagedThreadId 字段(仅填充 ClrInstanceIDOSThreadId)。你得靠 OSThreadId 关联 Windows ETW 的 ThreadID,再结合 ThreadStart 事件中的托管线程 ID 推断。

可行路径:

  • 同时订阅 Microsoft-Windows-DotNETRuntime/Thread/Start(事件 ID 260)和 ThreadPoolWorkerThreadStart(事件 ID 290),两者共享同一 OSThreadId
  • Thread/Start 事件中提取 ManagedThreadId,缓存到 Dictionary(key = OSThreadId),后续遇到 ThreadPoolWorkerThreadStart 就查表
  • 注意:此映射仅在该线程生命周期内有效;线程退出后需清理缓存,否则内存泄漏

实际排查并发瓶颈时,最易被忽略的是 ETW 事件的时间精度与 GC 暂停的干扰——比如 MonitorEnterStop 时间戳可能落在一次 GCStart 之后,导致你以为是锁等待,其实是 GC 抢占。务必打开 GC 关键字(0x1)并交叉比对。

相关专题

更多
session失效的原因
session失效的原因

session失效的原因有会话超时、会话数量限制、会话完整性检查、服务器重启、浏览器或设备问题等等。详细介绍:1、会话超时:服务器为Session设置了一个默认的超时时间,当用户在一段时间内没有与服务器交互时,Session将自动失效;2、会话数量限制:服务器为每个用户的Session数量设置了一个限制,当用户创建的Session数量超过这个限制时,最新的会覆盖最早的等等。

307

2023.10.17

session失效解决方法
session失效解决方法

session失效通常是由于 session 的生存时间过期或者服务器关闭导致的。其解决办法:1、延长session的生存时间;2、使用持久化存储;3、使用cookie;4、异步更新session;5、使用会话管理中间件。

733

2023.10.18

cookie与session的区别
cookie与session的区别

本专题整合了cookie与session的区别和使用方法等相关内容,阅读专题下面的文章了解更详细的内容。

88

2025.08.19

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

537

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

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

52

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

194

2025.08.29

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

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

387

2023.07.18

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

36

2026.01.14

热门下载

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

精品课程

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

共48课时 | 7.1万人学习

Excel 教程
Excel 教程

共162课时 | 11.7万人学习

PHP基础入门课程
PHP基础入门课程

共33课时 | 1.9万人学习

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

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