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("主线程:异步事件处理完成");
                });
            };
        }
相关推荐
xiaohe0726 分钟前
C#数据库操作系列---SqlSugar完结篇
网络·数据库·c#
yngsqq14 小时前
平面图环 内轮廓
c#
rockey62716 小时前
AScript之eval函数详解
c#·.net·script·eval·expression·动态脚本
He少年20 小时前
【AI 辅助案例分享】
人工智能·c#·编辑器·ai编程
工程师00721 小时前
栈和堆的概念
c#·栈和堆
不会编程的懒洋洋21 小时前
C# P/Invoke 基础
开发语言·c++·笔记·安全·机器学习·c#·p/invoke
Avalon7121 天前
Unity3D响应式渲染UI框架UniVue
游戏·ui·unity·c#·游戏引擎
njsgcs1 天前
solidworks折弯自动标注5 非90度折弯
c#·solidworks
狼与自由1 天前
clickhouse引擎
clickhouse·c#·linq
wangnaisheng1 天前
【C#】死锁详解:发生原因、优化解决方案
c#