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("主线程:异步事件处理完成");
                });
            };
        }
相关推荐
xiaoshuaishuai83 小时前
C# 接入 OpenClaw
windows·visualstudio·c#
gihigo19989 小时前
嵌入式幼儿园刷卡系统 (C#实现)
c#
qq_4542450310 小时前
通用引用管理框架
数据结构·架构·c#
aq553560010 小时前
三大编程语言深度对比:C# vs 易语言 vs 汇编
开发语言·汇编·c#
光泽雨10 小时前
c# 文件编译的过程
开发语言·c#
zxy284722530110 小时前
使用正运动的仿真软件C#
c#·仿真·运动控制·正运动·无硬件
三省持敬11 小时前
异步并发的“流量警察”:在C#中使用SemaphoreSlim进行并发控制的最佳实践
c#
唐青枫11 小时前
C#.NET IL 中间码 深入解析:从 C# 编译结果到 CLR 执行链路
c#·.net
xiaoshuaishuai812 小时前
C# 方言识别
开发语言·windows·c#
波波00714 小时前
写出稳定C#系统的关键:不可变性思想解析
开发语言·c#·wpf