0

0

c# 如何用 EventSource 和 EventListener 监控内部并发事件

煙雲

煙雲

发布时间:2026-01-15 09:58:47

|

168人浏览过

|

来源于php中文网

原创

eventsource 是 .net 轻量级事件发布机制,只负责“写”;eventlistener 是抽象基类,不可直接实例化,必须继承并重写 oneventsourcecreated 和 oneventwritten 才能接收事件。

c# 如何用 eventsource 和 eventlistener 监控内部并发事件

EventSource 是什么,为什么不能直接 new EventListener

EventSource 是 .NET 提供的轻量级、高性能事件发布机制,专为生产环境诊断设计;它本身不负责监听,只负责「写」。而 EventListener 是抽象基类,必须继承并重写 OnEventSourceCreatedOnEventWritten 才能接收事件——你不能直接 new EventListener(),否则毫无作用。

如何让自定义 EventListener 捕获并发相关事件(如 TaskScheduler、ThreadPool)

.NET 运行时自带多个内部 EventSource,比如 System.Threading.Tasks.TplEventSourceSystem.Runtime.ThreadPoolEventSource,它们默认是禁用的。要监听,必须在 OnEventSourceCreated 中显式启用对应事件源,并指定关键词(keyword)和等级(level)。

  • 关键词决定捕获哪类事件:例如 TaskScheduler 事件常用 EventSource.Settings.TaskScheduler(实际值为 0x10),ThreadPoolEventSource.Settings.ThreadPool(0x20)
  • 等级至少设为 EventLevel.Verbose 才能看到调度细节(如任务入队、线程唤醒)
  • 必须调用 EnableEvents(eventSource, level, keywords),且该调用需在 OnEventSourceCreated 内完成,延迟启用会丢事件
public class ConcurrencyEventListener : EventListener
{
    protected override void OnEventSourceCreated(EventSource eventSource)
    {
        if (eventSource.Name == "System.Threading.Tasks.TplEventSource")
        {
            EnableEvents(eventSource, EventLevel.Verbose, (EventKeywords)0x10); // TaskScheduler
        }
        else if (eventSource.Name == "System.Runtime.ThreadPoolEventSource")
        {
            EnableEvents(eventSource, EventLevel.Verbose, (EventKeywords)0x20); // ThreadPool
        }
    }

    protected override void OnEventWritten(EventWrittenEventArgs eventData)
    {
        if (eventData.EventName is "ScheduledTask" or "ThreadRequested" or "WorkerThreadStart")
        {
            Console.WriteLine($"[{eventData.EventSource.Name}] {eventData.EventName}: {string.Join(", ", eventData.PayloadNames.Zip(eventData.Payload, (n, v) => $"{n}={v}"))}");
        }
    }
}

常见漏掉的并发事件和调试陷阱

很多人以为启用了 TplEventSource 就能看见所有 Task 行为,但实际有三个关键限制:

Logomaster.ai
Logomaster.ai

Logo在线生成工具

下载
  • TaskCreationOptions.RunContinuationsAsynchronouslyTask.Run 创建的任务才触发 ScheduledTask;同步 continuation(如 ContinueWith 默认行为)不会发事件
  • ThreadPoolEventSourceThreadRequested 只在“线程不足需扩容”时发出,空闲线程复用过程无事件
  • 若程序启动后才创建 ConcurrencyEventListener 实例,此前已发生的调度事件(尤其是 AppDomain 初始化阶段的)完全丢失——必须尽早 new 并保持存活
  • 事件负载高时(如每秒数千任务),OnEventWritten 是同步调用,阻塞会导致事件被丢弃;生产环境应异步缓冲或限流处理

验证是否真收到了并发事件的最简方法

不要依赖日志滚动——写个可复现的最小触发片段,配合断点或计数器确认:

var listener = new ConcurrencyEventListener();
// 确保 listener 实例不被 GC(比如存为 static 字段)

Task.Run(() => Thread.Sleep(1)); // 触发 ThreadPool + Tpl 调度事件
Task.Factory.StartNew(() => { }, TaskCreationOptions.PreferFairness);

// 等待一小会儿再 Dispose,避免事件还在管道中
await Task.Delay(100);
listener.Dispose();

如果 OnEventWritten 完全没被调用,优先检查 EventSource.Name 是否拼错(大小写敏感)、.NET 版本是否支持(TplEventSource 在 .NET Core 2.1+ 稳定,.NET Framework 4.6.2+ 有但字段名不同)。

并发监控不是开箱即用的开关,每个 EventSource 都有自己隐含的触发边界和采样逻辑,想看到某类行为,得先确认它确实属于该事件源的输出范畴。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
线程和进程的区别
线程和进程的区别

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

723

2023.08.10

Swift iOS架构设计与MVVM模式实战
Swift iOS架构设计与MVVM模式实战

本专题聚焦 Swift 在 iOS 应用架构设计中的实践,系统讲解 MVVM 模式的核心思想、数据绑定机制、模块拆分策略以及组件化开发方法。内容涵盖网络层封装、状态管理、依赖注入与性能优化技巧。通过完整项目案例,帮助开发者构建结构清晰、可维护性强的 iOS 应用架构体系。

21

2026.03.03

C++高性能网络编程与Reactor模型实践
C++高性能网络编程与Reactor模型实践

本专题围绕 C++ 在高性能网络服务开发中的应用展开,深入讲解 Socket 编程、多路复用机制、Reactor 模型设计原理以及线程池协作策略。内容涵盖 epoll 实现机制、内存管理优化、连接管理策略与高并发场景下的性能调优方法。通过构建高并发网络服务器实战案例,帮助开发者掌握 C++ 在底层系统与网络通信领域的核心技术。

23

2026.03.03

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

77

2026.02.28

Golang 工程化架构设计:可维护与可演进系统构建
Golang 工程化架构设计:可维护与可演进系统构建

Go语言工程化架构设计专注于构建高可维护性、可演进的企业级系统。本专题深入探讨Go项目的目录结构设计、模块划分、依赖管理等核心架构原则,涵盖微服务架构、领域驱动设计(DDD)在Go中的实践应用。通过实战案例解析接口抽象、错误处理、配置管理、日志监控等关键工程化技术,帮助开发者掌握构建稳定、可扩展Go应用的最佳实践方法。

60

2026.02.28

Golang 性能分析与运行时机制:构建高性能程序
Golang 性能分析与运行时机制:构建高性能程序

Go语言以其高效的并发模型和优异的性能表现广泛应用于高并发、高性能场景。其运行时机制包括 Goroutine 调度、内存管理、垃圾回收等方面,深入理解这些机制有助于编写更高效稳定的程序。本专题将系统讲解 Golang 的性能分析工具使用、常见性能瓶颈定位及优化策略,并结合实际案例剖析 Go 程序的运行时行为,帮助开发者掌握构建高性能应用的关键技能。

48

2026.02.28

Golang 并发编程模型与工程实践:从语言特性到系统性能
Golang 并发编程模型与工程实践:从语言特性到系统性能

本专题系统讲解 Golang 并发编程模型,从语言级特性出发,深入理解 goroutine、channel 与调度机制。结合工程实践,分析并发设计模式、性能瓶颈与资源控制策略,帮助将并发能力有效转化为稳定、可扩展的系统性能优势。

26

2026.02.27

Golang 高级特性与最佳实践:提升代码艺术
Golang 高级特性与最佳实践:提升代码艺术

本专题深入剖析 Golang 的高级特性与工程级最佳实践,涵盖并发模型、内存管理、接口设计与错误处理策略。通过真实场景与代码对比,引导从“可运行”走向“高质量”,帮助构建高性能、可扩展、易维护的优雅 Go 代码体系。

20

2026.02.27

Golang 测试与调试专题:确保代码可靠性
Golang 测试与调试专题:确保代码可靠性

本专题聚焦 Golang 的测试与调试体系,系统讲解单元测试、表驱动测试、基准测试与覆盖率分析方法,并深入剖析调试工具与常见问题定位思路。通过实践示例,引导建立可验证、可回归的工程习惯,从而持续提升代码可靠性与可维护性。

4

2026.02.27

热门下载

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

精品课程

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

共578课时 | 76.6万人学习

国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1万人学习

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

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