C#中发布订阅的阻塞非阻塞

在 C# 中,事件的发布和订阅机制遵循以下规则:

1. 默认行为:同步执行,阻塞发布线程

如果事件订阅者(主线程)的处理方法是同步的,子线程在发布事件后会等待所有订阅者执行完毕,才会继续执行后续代码。

Block_Test.cs

cs 复制代码
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Threading;
namespace WindowsFormsApp1
{
    public class Block_Test
    {

        // 定义一个委托类型
        public delegate void MyEventHandler(object sender, EventArgs e);

        // 定义一个事件
        public event MyEventHandler MyEvent;

        // 触发事件的方法
        public void TriggerEvent()
        {
            Console.WriteLine($"thread--{Thread.CurrentThread.ManagedThreadId}--");
            Console.WriteLine("事件触发前");

            // 触发事件,会调用所有注册的处理程序
            MyEvent?.Invoke(this, EventArgs.Empty);

            Console.WriteLine("事件触发后");
        }
    }
}

program.cs

cs 复制代码
         public Form1()
        {
            InitializeComponent();
            
            

            

            Console.WriteLine($"Form1--{Thread.CurrentThread.ManagedThreadId}--");
            signalThread = new Thread(publisher.TriggerEvent);
            signalThread.IsBackground = true; // 设置为后台线程
            
            
            //OnEventHandlerAsync();//异步订阅(不阻塞发布线程)
            // 注册事件处理程序
            publisher.MyEvent += Program_MyEvent;//阻塞发布线程


            // 触发事件
            //program.TriggerEvent();
        }
        public void Program_MyEvent(object sender, EventArgs e)
        {
            
            //下面的代码发布线程会阻塞
            if (this.InvokeRequired)  // 判断是否需要切换线程
            {
                // 需要切换到主线程
                this.Invoke(new EventHandler(Program_MyEvent), sender, e);
                return;
            }

            Console.WriteLine($"mian--{Thread.CurrentThread.ManagedThreadId}--");
            Console.WriteLine("事件处理程序执行中");
            // 模拟一些工作
            System.Threading.Thread.Sleep(10000);
            this.button1.Text = "button2";
            Console.WriteLine("事件处理程序执行完毕");
            
        }

2. 异步执行:发布后立即继续(不阻塞)

如果需要子线程在发布事件后立即继续执行,可以在订阅时使用异步处理

cs 复制代码
        public Form1()
        {
            InitializeComponent();
            
            

            Console.WriteLine($"Form1--{Thread.CurrentThread.ManagedThreadId}--");
            signalThread = new Thread(publisher.TriggerEvent);
            signalThread.IsBackground = true; // 设置为后台线程
            
            
            OnEventHandlerAsync();//异步订阅(不阻塞发布线程)
            // 注册事件处理程序
            //publisher.MyEvent += Program_MyEvent;//阻塞发布线程


            // 触发事件
            //program.TriggerEvent();
        }
        private void OnEventHandlerAsync()
        {
            // 异步订阅(不阻塞发布线程)
            publisher.MyEvent += async (sender, e) =>
            {
                await Task.Run(() =>
                {
                    Console.WriteLine("主线程:异步处理事件(模拟耗时操作)");
                    Thread.Sleep(10000);
                    Console.WriteLine("主线程:异步事件处理完成");
                });
            };
        }
相关推荐
懒人Ethan14 小时前
解决一个C# 在Framework 4.5反序列化的问题
java·前端·c#
mysolisoft15 小时前
Avalonia+ReactiveUI实现记录自动更新
c#·avalonia·reactiveui·sourcegenerator
心疼你的一切16 小时前
使用Unity引擎开发Rokid主机应用的模型交互操作
游戏·ui·unity·c#·游戏引擎·交互
韩立学长17 小时前
【开题答辩实录分享】以《C#大型超市商品上架调配管理系统的设计与实现》为例进行答辩实录分享
开发语言·c#
玩泥巴的19 小时前
.NET驾驭Word之力:数据驱动文档 - 邮件合并与自定义数据填充完全指南
c#·word·.net·com互操作
心疼你的一切1 天前
使用Unity引擎开发Rokid主机应用的全面配置交互操作
学习·游戏·unity·c#·游戏引擎·交互
椒颜皮皮虾྅1 天前
【DeploySharp 】基于DeploySharp 的深度学习模型部署测试平台:安装和使用流程
人工智能·深度学习·开源·c#·openvino
kalvin_y_liu1 天前
【MES架构师与C#高级工程师(设备控制方向)两大职业路径的技术】
开发语言·职场和发展·c#·mes
椒颜皮皮虾2 天前
基于DeploySharp 的深度学习模型部署测试平台:支持YOLO全系列模型
c#
李宥小哥2 天前
C#基础10-结构体和枚举
java·开发语言·c#