详解 C# 中基于发布-订阅模式的 Messenger 消息传递机制:Messenger.Default.Send/Register
🧑 博主简介:CSDN博客专家、CSDN平台优质创作者,高级开发工程师,数学专业,10年以上C/C++, C#, Java等多种编程语言开发经验,拥有高级工程师证书;擅长C/C++、C#等开发语言,熟悉Java常用开发技术,能熟练应用常用数据库SQL server,Oracle,mysql,postgresql等进行开发应用,熟悉DICOM医学影像及DICOM协议,业余时间自学JavaScript,Vue,qt,python等,具备多种混合语言开发能力。撰写博客分享知识,致力于帮助编程爱好者共同进步。欢迎关注、交流及合作,提供技术支持与解决方案。
技术合作请加本人wx(注明来自csdn):xt20160813
详解 C# 中基于发布-订阅模式的 Messenger 消息传递机制:Messenger.Default.Send/Register
在 C# 开发中,特别是在 MVVM(Model-View-ViewModel)架构的客户端应用程序中,跨组件通信是实现模块化设计和松耦合的关键。Messenger
是一种基于发布-订阅模式(Publish-Subscribe Pattern)的消息传递机制,广泛应用于 WPF、WinForms 和其他 C# 框架中。本文以 Messenger.Default.Send
和 Messenger.Default.Register
为核心,结合历史对话中提到的代码示例(如 Messenger.Default.Send<string>("Recovery", "fanxuan")
),详细解析其实现原理、应用场景、技术细节,并提供完整代码示例和开发注意事项。文章特别融入国产化环境(如银河麒麟系统)的开发视角,确保内容准确、实用。
一、引言
发布-订阅模式是一种设计模式,允许消息发布者(Publisher)将消息发送给订阅者(Subscriber),而无需直接耦合。Messenger
是 MVVM 框架(如 MVVM Light、Prism 或自定义实现)中常见的消息传递工具,通过 Messenger.Default.Send
和 Messenger.Default.Register
实现异步、定向的模块间通信。
在历史对话中,代码片段展示了通过 Messenger.Default.Send<string>("Recovery", "fanxuan")
发送消息,并在接收端通过 Messenger.Default.Register<string>
捕获消息并触发 button1_Click
方法。这种机制广泛用于 UI 状态管理(如恢复按钮状态)、跨 ViewModel 通信或后台服务协调。本文将从原理、实现、代码示例到国产化开发适配,全面解析这一机制。
二、发布-订阅模式与 Messenger 概述
1. 发布-订阅模式
发布-订阅模式的核心思想是:
- 发布者:发送消息但不关心谁接收。
- 订阅者:注册对特定消息的兴趣,接收并处理消息。
- 消息中介:管理消息的分发,通常通过令牌(Token)或消息类型过滤。
优点:
- 松耦合:发布者和订阅者无需直接引用,降低模块间依赖。
- 灵活性:支持一对多通信,动态注册/注销订阅者。
- 线程安全:常结合 Dispatcher 机制,确保 UI 操作在主线程执行。
2. Messenger 的功能
Messenger
是 MVVM 框架中的消息传递工具(如 MVVM Light 的 GalaSoft.MvvmLight.Messaging.Messenger
),提供以下功能:
- 发送消息:通过
Messenger.Default.Send<T>(message, token)
发送特定类型的消息。 - 注册订阅:通过
Messenger.Default.Register<T>(recipient, token, action)
注册消息处理逻辑。 - 令牌机制:使用
token
(如字符串"fanxuan"
)实现定向消息传递,减少无关模块的处理开销。 - 线程调度:结合
Dispatcher
或异步机制,确保线程安全。
在历史对话中,Messenger.Default.Send<string>("Recovery", "fanxuan")
表示发送字符串消息 "Recovery"
,目标是注册了 "fanxuan"
令牌的订阅者。
三、Messenger 的核心方法
1. Messenger.Default.Send
签名:
void Send<T>(T message, object token = null);
- 参数:
T message
:消息内容,支持任意类型(如string
、自定义类)。token
:可选的令牌,用于过滤接收者,通常为字符串或对象。
- 作用:将消息广播给所有匹配
token
和消息类型T
的订阅者。 - 示例:
// 发送字符串消息 "Recovery",令牌为 "fanxuan" Messenger.Default.Send<string>("Recovery", "fanxuan");
2. Messenger.Default.Register
签名:
void Register<T>(object recipient, object token, Action<T> action);
- 参数:
recipient
:订阅者对象(通常为this
,表示当前类实例)。token
:与发送端匹配的令牌。action
:收到消息后执行的回调函数,接收消息内容T
。
- 作用:注册一个消息处理逻辑,仅处理匹配
token
和类型T
的消息。 - 示例:
// 注册处理 "fanxuan" 令牌的字符串消息 Messenger.Default.Register<string>(this,"fanxuan",message => { Console.WriteLine($"Received: {message}"); } );
3. 消息传递流程
以下是 Messenger
的工作流程(结合历史对话中的代码):
graph LRA[Send: Messenger.Default.Send<string>("Recovery", "fanxuan")] --> B[Messenger 中介]B --> C[查找匹配 "fanxuan" 和 string 的订阅者]C --> D[Register: Messenger.Default.Register<string>(this, "fanxuan", ...)]D --> E[执行回调: DicomOperateDispatcher.Invoke(button1_Click)]
四、Messenger 的实现原理
Messenger
通常基于单例模式(Messenger.Default
),内部维护一个订阅者列表。核心实现逻辑如下:
-
消息注册:
Register
方法将订阅者的{recipient, token, action}
组合存储在一个字典或列表中。- 键为消息类型
T
和token
,值为回调函数Action<T>
。
-
消息发送:
Send
方法遍历订阅者列表,查找匹配T
和token
的注册项。- 对每个匹配的订阅者,调用其
action
回调,传递消息内容。
-
线程安全:
Messenger
通常不直接处理线程切换,需结合Dispatcher
(如 WPF 的Dispatcher.Invoke
)确保 UI 操作在主线程执行。
-
令牌机制:
- 令牌(
token
)通过哈希比较(如字符串的Equals
方法)实现高效匹配。 - 如果
token
为null
,消息广播给所有订阅了类型T
的接收者。
- 令牌(
五、Messenger 在历史对话中的应用
在历史对话中,代码展示了 Messenger
在医学影像系统(DICOM 相关)中的应用,用于触发恢复按钮状态:
发送端:
case ItCallBackMSG.Recovery: // 复原按钮状态通过 ESC 键触发
{// 发送恢复命令Messenger.Default.Send<string>("Recovery", "fanxuan");break;
}
- 场景:当用户按下 ESC 键,系统触发
ItCCallBackMSG.Recovery
状态,发送"Recovery"
消息,目标为"fanxuan"
通道。 - 目的:通知订阅了
"fanxuan"
的模块执行状态恢复(如恢复 UI 按钮或 DICOM 数据状态)。
接收端:
Messenger.Default.Register<string>(this,"fanxuan",message => DicomOperateDispatcher.Invoke(new Onclick(button1_Click), null, null)
);
- 场景:某个类(可能是 View 或 ViewModel)注册了
"fanxuan"
通道的字符串消息处理。 - 处理逻辑:收到消息后,通过
DicomOperateDispatcher.Invoke
调用button1_Click
方法,可能是恢复 UI 按钮状态或重置 DICOM 数据。 - 线程安全:
DicomOperateDispatcher.Invoke
确保button1_Click
在 UI 线程执行,避免跨线程访问冲突。
分析:
- 消息内容:
"Recovery"
表示恢复命令,可能触发 UI 重置、数据恢复或其他状态复原。 - 令牌:
"fanxuan"
(可能意为“反选”或特定模块标识)确保消息只发送给特定订阅者。 - 问题:当前代码未对
message
内容进行判断,可能响应所有"fanxuan"
通道的消息,建议添加条件判断(如if (message == "Recovery")
)。
六、完整代码示例
以下是一个完整的 C# 示例,展示 Messenger
的发送和接收过程,结合 WPF 和 MVVM 架构,模拟恢复按钮状态的场景。
1. 安装 MVVM Light
在项目中添加 MVVM Light NuGet 包:
Install-Package MvvmLight
2. 发送端(MainViewModel.cs)
using GalaSoft.MvvmLight;
using GalaSoft.MvvmLight.Messaging;public class MainViewModel : ViewModelBase
{public MainViewModel(){// 模拟 ESC 键触发恢复SendRecoveryCommand();}private void SendRecoveryCommand(){// 发送恢复消息,目标为 "fanxuan" 通道Messenger.Default.Send<string>("Recovery", "fanxuan");}
}
3. 接收端(ButtonViewModel.cs)
using GalaSoft.MvvmLight;
using GalaSoft.MvvmLight.Messaging;
using System.Windows;public class ButtonViewModel : ViewModelBase
{public ButtonViewModel(){// 注册 "fanxuan" 通道的字符串消息Messenger.Default.Register<string>(this,"fanxuan",message =>{// 仅处理 "Recovery" 消息if (message == "Recovery"){// 在 UI 线程执行按钮状态恢复Application.Current.Dispatcher.Invoke(() =>{ButtonClickHandler();});}});}private void ButtonClickHandler(){// 模拟恢复按钮状态MessageBox.Show("Button state recovered!");// 实际逻辑:重置 UI 按钮、DICOM 数据状态等}
}
4. 主程序(App.xaml.cs)
using GalaSoft.MvvmLight.Messaging;public partial class App : Application
{protected override void OnStartup(StartupEventArgs e){base.OnStartup(e);// 初始化 ViewModelvar mainViewModel = new MainViewModel();var buttonViewModel = new ButtonViewModel();}
}
运行结果:
- 当
MainViewModel
发送"Recovery"
消息时,ButtonViewModel
捕获消息并执行ButtonClickHandler
,显示提示框。 Dispatcher.Invoke
确保 UI 操作在主线程执行。
七、国产化环境中的应用
结合历史对话中关于银河麒麟系统的离线部署经验,以下是 Messenger
在国产化环境中的开发注意事项:
-
离线安装 MVVM Light:
- 在银河麒麟上离线安装 NuGet 包:
sudo rpm -ivh mvvmlightlibs-5.4.1.rpm --nodeps --force
- 确保项目引用
GalaSoft.MvvmLight.dll
。
- 在银河麒麟上离线安装 NuGet 包:
-
DICOM 集成:
Messenger
可用于 DICOM 系统中模块间通信(如 UI 与 PACS 数据同步)。- 示例:解析 DICOM 文件并发送状态更新:
using pydicom; // 需通过 Python 脚本解析后传递 var ds = pydicom.dcmread("mri.dcm"); Messenger.Default.Send<string>("DICOMLoaded", "fanxuan");
-
防火墙配置:
- 如果
Messenger
用于跨进程通信(如与 Orthanc PACS 服务交互),需放行端口:sudo firewall-cmd --zone=public --add-port=4242/tcp --permanent sudo firewall-cmd --reload
- 如果
-
Fitten Code 集成:
- 使用 Fitten Code 生成消息处理代码:
// @Fitten 生成 Messenger 注册代码 Messenger.Default.Register<string>(this,"fanxuan",message => { if (message == "Recovery") { /* 恢复逻辑 */ } } );
- 使用 Fitten Code 生成消息处理代码:
八、常见问题与解决方案
问题 | 解决方案 |
---|---|
消息未被接收 | 检查 Register 和 Send 的 token 和消息类型是否匹配。 |
UI 线程冲突 | 使用 Dispatcher.Invoke 确保回调在主线程执行。 |
国产系统依赖缺失 | 离线安装 MVVM Light:sudo rpm -ivh mvvmlightlibs.rpm --nodeps 。 |
消息广播过广 | 添加 token (如 "fanxuan" )或条件判断(如 if (message == "Recovery") )。 |
九、总结
Messenger.Default.Send
和 Messenger.Default.Register
是 C# 中基于发布-订阅模式的强大消息传递机制,适用于 MVVM 架构中的模块间通信。核心特点:
- 松耦合:通过
token
实现定向消息传递。 - 线程安全:结合
Dispatcher
确保 UI 操作安全。 - 灵活性:支持任意消息类型和动态注册/注销。
结合历史对话中的 DICOM 系统场景,Messenger
在恢复按钮状态、跨模块通信中发挥了关键作用。在国产化环境(如银河麒麟)中,需注意离线依赖管理和防火墙配置。提供的代码示例可直接运行,适合开发者快速上手。
建议:
- 在接收端添加消息内容判断,提升代码健壮性。
- 使用 MVVM Light 或 Prism 的
Messenger
实现标准化开发。 - 在国产化环境中测试消息传递的性能和稳定性。
参考资料
- MVVM Light 文档:http://www.mvvmlight.net/
- DICOM 标准:http://dicom.nema.org/
- C# 委托与事件:https://docs.microsoft.com/en-us/dotnet/csharp/programming-guide/events/
相关文章:

详解 C# 中基于发布-订阅模式的 Messenger 消息传递机制:Messenger.Default.Send/Register
🧑 博主简介:CSDN博客专家、CSDN平台优质创作者,高级开发工程师,数学专业,10年以上C/C, C#, Java等多种编程语言开发经验,拥有高级工程师证书;擅长C/C、C#等开发语言,熟悉Java常用开…...

多场景游戏AI新突破!Divide-Fuse-Conquer如何激发大模型“顿悟时刻“?
多场景游戏AI新突破!Divide-Fuse-Conquer如何激发大模型"顿悟时刻"? 大语言模型在强化学习中偶现的"顿悟时刻"引人关注,但多场景游戏中训练不稳定、泛化能力差等问题亟待解决。Divide-Fuse-Conquer方法,通过…...

Java 函数式接口(Functional Interface)
一、理论说明 1. 函数式接口的定义 Java 函数式接口是一种特殊的接口,它只包含一个抽象方法(Single Abstract Method, SAM),但可以包含多个默认方法或静态方法。函数式接口是 Java 8 引入 Lambda 表达式的基础,通过函…...

分布式锁总结
文章目录 分布式锁什么是分布式锁?分布式锁的实现方式基于数据库(mysql)实现基于缓存(redis)多实例并发访问问题演示项目代码(使用redis)配置nginx.confjmeter压测复现问题并发是1,即不产生并发问题并发30测试,产生并发问题(虽然单实例是synchronized&am…...

使用MybatisPlus实现sql日志打印优化
背景: 在排查无忧行后台服务日志时,一个请求可能会包含多个执行的sql,经常会遇到SQL语句与对应参数不连续显示,或者参数较多需要逐个匹配的情况。这种情况下,如果需要还原完整SQL语句就会比较耗时。因此,我…...
springboot中redis的事务的研究
redis的事务类似于队列操作,执行过程分为三步: 开启事务入队操作执行事务 使用到的几个命令如下: 命令说明multi开启一个事务exec事务提交discard事务回滚watch监听key(s):当监听一个key(s)时,如果在本次事务提交之…...
为什么我输入对了密码,还是不能用 su 切换到 root?
“为什么我输入对了密码,还是不能用 su 切换到 root?” 其实这背后可能不是“密码错了”,而是系统不允许你用 su 切 root,即使密码对了。 👇 以下是最常见的几个真正原因: ❌ 1. Root 用户没有设置密码&…...

client.chat.completions.create方法参数详解
response client.chat.completions.create(model"gpt-3.5-turbo", # 必需参数messages[], # 必需参数temperature1.0, # 可选参数max_tokensNone, # 可选参数top_p1.0, # 可选参数frequency_penalty0.0, # 可选参数presenc…...
量子计算与云计算的融合:技术前沿与应用前景
目录 引言 量子计算基础 量子计算的基本原理 量子计算的优势与挑战 量子计算的发展阶段 云计算基础 云计算的基本概念 云计算的应用领域 云计算面临的挑战 量子计算与云计算的结合 量子云计算的概念与架构 量子云计算的服务模式 量子云计算的优势 量子云计算的发展…...
《企业级日志该怎么打?Java日志规范、分层设计与埋点实践》
大家好呀!👋 今天我们要聊一个Java开发中超级重要但又经常被忽视的话题——日志系统!📝 不管你是刚入门的小白,还是工作多年的老司机,日志都是我们每天都要打交道的"好朋友"。那么,如…...
python模块管理环境变量
概要 在 Python 应用中,为了将配置信息与代码分离、增强安全性并支持多环境(开发、测试、生产)运行,使用专门的模块来管理环境变量是最佳实践。常见工具包括: 标准库 os.environ:直接读取操作系统环境变量…...
【泛微系统】后端开发Action常用方法
后端开发Action常用方法 代码实例经验分享:代码实例 经验分享: 本文分享了后端开发中处理工作流Action的常用方法,主要包含以下内容:1) 获取工作流基础信息,如流程ID、节点ID、表单ID等;2) 操作请求信息,包括请求紧急程度、操作类型、用户信息等;3) 表单数据处理,展示…...
【算法】力扣体系分类
第一章 算法基础题型 1.1 排序算法题 1.1.1 冒泡排序相关题 冒泡排序是一种简单的排序算法,它重复地走访过要排序的数列,一次比较两个元素,如果它们的顺序错误就把它们交换过来。走访数列的工作是重复地进行直到没有再需要交换,…...
sql:如何查询一个数据表字段:Scrp 数据不为空?
在SQL中,要查询一个数据表中的字段 Scrp 不为空的记录,可以使用 IS NOT NULL 条件。以下是一个基本的SQL查询示例: SELECT * FROM your_table_name WHERE Scrp IS NOT NULL;在这个查询中,your_table_name 应该替换为你的实际数据…...

深入浅出人工智能:机器学习、深度学习、强化学习原理详解与对比!
各位朋友,大家好!今天咱们聊聊人工智能领域里最火的“三剑客”:机器学习 (Machine Learning)、深度学习 (Deep Learning) 和 强化学习 (Reinforcement Learning)。 听起来是不是有点高大上? 别怕,我保证把它们讲得明明…...
索引下探(Index Condition Pushdown,简称ICP)
索引下探(Index Condition Pushdown,简称ICP)是一种数据库查询优化技术,常见于MySQL等关系型数据库中。 1. 核心概念 作用:将原本在服务器层执行的WHERE条件判断尽可能下推到存储引擎层执行。减少回表查询次数支持部…...

基于 ColBERT 框架的后交互 (late interaction) 模型速递:Reason-ModernColBERT
一、Reason-ModernColBERT 模型概述 Reason-ModernColBERT 是一种基于 ColBERT 框架的后交互 (late interaction) 模型,专为信息检索任务中的推理密集型场景设计。该模型在 reasonir-hq 数据集上进行训练,于 BRIGHT 基准测试中取得了极具竞争力的性能表…...

vector中reserve导致的析构函数问题
接上一节vector实现,解决杨辉三角问题时,我在最后调试的时候,发现return vv时,调用析构函数,到第四步时才析构含有14641的vector。我设置了一个全局变量i来记录。 初始为35: 当为39时,也就是第…...

微软开源多智能体自定义自动化工作流系统:构建企业级AI驱动的智能引擎
微软近期推出了一款开源解决方案加速器——Multi-Agent Custom Automation Engine Solution Accelerator,这是一个基于AI多智能体协作的自动化工作流系统。该系统通过指挥多个智能体(Agent)协同完成复杂任务,显著提升企业在数据处理、业务流程管理等场景中的效率与准确性。…...
关于vector、queue、list哪边是front、哪边是back,增加、删除元素操作
容器的 front、back 及操作方向 1.1vector(动态数组) 结构:连续内存块,支持快速随机访问。 操作方向: front:第一个元素(索引 0)。 back:最后一个元素(索引…...
KubeVela入门到精通-K8S多集群交付
目录 1、介绍 2、部署 3、部署UI界面 4、御载 5、Velaux概念 6、OAM应用模型介绍 7、应用部署计划 8、系统架构 9、基础环境配置 9.1 创建项目 9.2 创建集群 9.3 创建交付目标 9.4 创建环境 9.5、创建服务测试 9.6、服务操作 10、插件、项目、权限管理 10.1 插…...
RocketMq的消息类型及代码案例
RocketMQ 提供了多种消息类型,以满足不同业务场景对 顺序性、事务性、时效性 的要求。其核心设计思想是通过解耦 “消息传递模式” 与 “业务逻辑”,实现高性能、高可靠的分布式通信。 一、主要类型包括 普通消息(基础类型)顺序…...
Eigen 直线拟合/曲线拟合/圆拟合/椭圆拟合
一、直线拟合 使用Eigen库进行直线拟合是数据分析和科学计算中的常见任务,主要通过最小二乘法实现。以下是关键实现方法和示例: 核心原理最小二乘法通过最小化点到直线距离的平方和来求解最优直线参数间接平差法是最小二乘法的具体实现形式,适用于直线拟合场景通过构建误差…...

安卓无障碍脚本开发全教程
文章目录 第一部分:无障碍服务基础1.1 无障碍服务概述核心功能: 1.2 基本原理与架构1.3 开发环境配置所需工具:关键依赖: 第二部分:创建基础无障碍服务2.1 服务声明配置2.2 服务配置文件关键属性说明: 2.3 …...
svn迁移到git保留记录和Python字符串格式化 f-string的进化历程
svn迁移到git保留记录 and Python字符串格式化(二): f-string的进化历程 在将项目从SVN迁移到Git时,保留完整的版本历史记录非常重要。下面是详细的步骤和工具,可以帮助你完成这一过程: 安装Git和SVN工具 首先&#…...

SOC-ESP32S3部分:10-GPIO中断按键中断实现
飞书文档https://x509p6c8to.feishu.cn/wiki/W4Wlw45P2izk5PkfXEaceMAunKg 学习了GPIO输入和输出功能后,参考示例工程,我们再来看看GPIO中断,IO中断的配置分为三步 配置中断触发类型安装中断服务注册中断回调函数 ESP32-S3的所有通用GPIO…...
【神经网络与深度学习】扩散模型之原理解释
引言: 在人工智能的生成领域,扩散模型(Diffusion Model)是一项极具突破性的技术。它不仅能够生成高质量的图像,还可以应用于音频、3D建模等领域。扩散模型的核心思想来源于物理扩散现象,其工作方式类似于从…...
语音合成之十六 语音合成(TTS)跳跃与重复问题的解析:成因、机制及解决方案
语音合成(TTS)跳跃与重复问题的解析:成因、机制及解决方案 引言TTS中跳跃与重复问题的根本原因注意力机制的失效文本到语音的对齐挑战自回归(AR)TTS模型的固有挑战时长建模的重要性输入数据特征的影响 构建鲁棒性&…...

战略-2.1 -战略分析(PEST/五力模型/成功关键因素)
战略分析路径,先宏观(PEST)、再产业(产品生命周期、五力模型、成功关键因素)、再竞争对手分析、最后企业内部分析。 本文介绍:PEST、产品生命周期、五力模型、成功关键因素、产业内的战略群组 一、宏观环境…...

python第三方库安装错位
问题所在 今天在安装我的django库时,我的库安装到了python3.13版本。我本意是想安装到python3.11版本的。我的pycharm右下角也设置了python3.11 但是太可恶了,我在pycharm的项目终端执行安装命令的时候还是给我安装到了python3.13的位置。 解决方法 我…...