redis pub/sub 是跨进程/网络的异步消息广播,c# 事件是进程内同步委托回调;前者天然分布式、发后即忘,后者纯本地、同步执行且不保证可靠性。

Redis Pub/Sub 和 C# 事件本质不同:一个是跨进程/网络的异步消息广播,一个是进程内同步(或可控异步)的委托回调。 别把 event 当成“轻量版 Redis 订阅”——它们解决的问题域、生命周期、可靠性边界完全不在一个层面。
订阅者能不能跨进程?这是最根本的分水岭
C# 事件(event)只在当前 AppDomain 或同一进程内有效。两个独立运行的 .NET 进程(比如 WebAPI 和后台 Worker),即使代码一模一样,MyClass.SomeEvent += handler 也完全收不到对方触发的事件——没网络、没序列化、没中间件,纯内存引用。
而 Redis Pub/Sub 的订阅者可以是任意语言、任意机器上的客户端:SUBSCRIBE order.created 后,只要连的是同一个 Redis 实例,Java 服务、Python 脚本、Node.js 管理后台都能实时收到消息。
- ✅ Redis Pub/Sub:天然分布式,适合微服务间松耦合通信
- ❌ C# 事件:纯本地,连跨线程都得自己加
Task.Run或调度器,更别说跨进程
消息丢了怎么办?可靠性设计逻辑完全不同
Redis Pub/Sub 是“发后即忘(fire-and-forget)”:发布者调用 PUBLISH 成功,Redis 就把消息推给当时在线的所有订阅者;如果订阅者掉线了,消息直接丢弃,不重试、不持久化、不存 backlog。
C# 事件则完全相反:触发 SomeEvent?.Invoke() 时,所有已注册的委托会**同步执行**(除非你手动扔进线程池)。它不关心“送达”,只保证“当前注册者立刻被调用”——但这也意味着,如果某个 handler 抛异常,整个事件链可能中断(除非你用 GetInvocationList() 手动遍历并 try-catch)。
- ⚠️ Redis Pub/Sub 不适合任务队列场景(比如下单后发邮件),要用
Redis Streams或 RabbitMQ - ⚠️ C# 事件里写耗时操作(如 HTTP 调用)会阻塞发布者线程,必须显式异步包装
怎么在 C# 里真正用好 Redis Pub/Sub?别直接裸写 StackExchange.Redis
直接用 IDatabase.PublishAsync() 和 IBasicClient.Subscribe() 容易踩坑:连接断开不自动重连、订阅丢失无感知、消息反序列化硬编码、线程上下文错乱(比如在 ASP.NET Core 中订阅后试图更新 UI 控件)。
推荐做法:
- 用
ISubscriber单例 +ConnectionMultiplexer自动重连机制,避免每次新建连接 - 消息体统一走 JSON 序列化(如
System.Text.Json),字段加版本号,避免前后端结构不一致 - 业务逻辑不要写在订阅回调里,而是转发到
BackgroundService或 MediatR 的IRequestHandler中处理
var subscriber = _connection.GetSubscriber();
await subscriber.SubscribeAsync("user.registered", async (channel, message) =>
{
var evt = JsonSerializer.Deserialize<UserRegisteredEvent>(message);
// ✅ 转发给后台服务,不阻塞 Redis 回调线程
await _backgroundQueue.EnqueueAsync(() => HandleUserRegistered(evt));
});最常被忽略的一点:Redis Pub/Sub 的频道名(channel)不是命名空间,它不支持层级继承或通配符继承——PSUBSCRIBE user.* 可以匹配 user.registered,但 SUBSCRIBE user 和 SUBSCRIBE user.registered 是两个完全无关的频道。C# 事件的命名(user.Registered)看起来像层级,其实是纯符号,和 Redis 频道语义毫无关系。










