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("主线程:异步事件处理完成");
                });
            };
        }
相关推荐
脑电信号要分类3 小时前
将多张图片拼接成一个pdf文件输出
pdf·c#·apache
njsgcs4 小时前
c# solidworks 折弯系数检查
开发语言·c#
格林威5 小时前
工业相机图像采集:Grab Timeout 设置建议——拒绝“假死”与“丢帧”的黄金法则
开发语言·人工智能·数码相机·计算机视觉·c#·机器视觉·工业相机
唐青枫5 小时前
C#.NET SignalR + Redis Backplane 深入解析:多节点部署与跨实例消息同步
c#·.net
FL162386312918 小时前
[C#][winform]segment-anything分割万物部署onnx模型一键抠图演示
开发语言·c#
love530love20 小时前
OpenClaw 手机直连配置全流程
人工智能·windows·python·智能手机·c#·agent·openclaw
bcbobo21cn21 小时前
C# byte类型和byte数组的使用
开发语言·c#·字节数组·byte类型
月巴月巴白勺合鸟月半1 天前
一次PDF文件的处理(一)
pdf·c#
大鹏说大话1 天前
Java 锁膨胀机制深度解析:从偏向锁到重量级锁的进化之路
开发语言·c#
武藤一雄1 天前
WPF处理耗时操作的7种方法
microsoft·c#·.net·wpf