0

0

Golang微服务中如何使用Redis进行服务间通信

P粉602998670

P粉602998670

发布时间:2026-01-07 07:49:10

|

151人浏览过

|

来源于php中文网

原创

不能,Redis Pub/Sub 不适合微服务间可靠通信。它是发即忘机制,无持久化、无ACK、订阅者离线消息丢失,仅适用于低频广播场景如配置刷新;可靠通信应选Kafka/RabbitMQ或Redis Streams。

golang微服务中如何使用redis进行服务间通信

Redis Pub/Sub 能否用于微服务间可靠通信?

不能直接用作生产级服务间通信。Redis 的 PUB/SUB 是「发即忘」机制:消息不持久、无 ACK、订阅者离线期间消息全丢,且不支持多消费者组语义。它适合广播通知(如配置刷新、缓存失效),但不适合订单创建、支付回调这类需要至少一次投递的业务场景。

  • 常见错误现象:SUBSCRIBE 后收不到消息 → 检查是否在 redis-cli 中用了 MONITOR 占用连接,或 Go 客户端未保持长连接
  • 使用场景限制:仅适用于低频、容忍丢失、无需顺序保障的事件,比如「清空某类缓存」
  • 替代方案优先级:需要可靠性 → 用 Kafka / RabbitMQ;轻量级且能接受一定风险 → Redis Streams(支持消费者组、ACK、消息重试)

用 Redis Streams 实现带确认的服务间消息传递

Redis Streams 是更接近消息队列的结构,支持消费者组(Consumer Group)、消息 ID 自增、ACKpending list,可模拟至少一次语义。

package main

import ( "context" "fmt" "time" "github.com/go-redis/redis/v8" )

var rdb = redis.NewClient(&redis.Options{ Addr: "localhost:6379", })

func sendOrderEvent() { ctx := context.Background() _, err := rdb.XAdd(ctx, &redis.XAddArgs{ Stream: "order_stream", Values: map[string]interface{}{"order_id": "12345", "status": "created"}, }).Result() if err != nil { panic(err) } }

func consumeOrderEvents() { ctx := context.Background() // 创建消费者组(仅需执行一次) rdb.XGroupCreateMkStream(ctx, "order_stream", "order_service_group", "$").Err()

for {
    msgs, err := rdb.XReadGroup(ctx, &redis.XReadGroupArgs{
        Group:    "order_service_group",
        Consumer: "svc-a-01",
        Streams:  []string{"order_stream", ">"},
        Count:    1,
        Block:    100 * time.Millisecond,
    }).Result()
    if err != nil && err != redis.Nil {
        fmt.Printf("read error: %v\n", err)
        continue
    }
    if len(msgs) == 0 {
        continue
    }

    for _, msg := range msgs[0].Messages {
        fmt.Printf("received: %v\n", msg.Values)
        // 处理成功后手动 ACK
        rdb.XAck(ctx, "order_stream", "order_service_group", msg.ID)
        // 从 pending list 中移除(可选,ACK 后自动清理)
    }
}

}

  • 关键参数:Streams: []string{"order_stream", ">"} 中的 > 表示只读取新消息;首次消费用 0 可回溯历史
  • 消费者组名(如 order_service_group)必须全局唯一,不同微服务应使用不同组名,否则会争抢同一条消息
  • 每个消费者实例需有唯一 Consumer 名(如 svc-a-01),便于排查 pending 消息归属
  • 性能注意:频繁调用 XAck 不影响吞吐,但若处理逻辑失败又未 NAK(Redis 无原生 NAK,需用 XClaim 抢回重试),pending 消息会堆积

如何避免多个服务重复消费同一份数据?

根本上不是靠 Redis 隔离,而是靠「服务职责划分 + 幂等设计」。Redis Streams 本身不阻止多组消费,但你可以通过命名和路由策略控制:

易优cms汽车车辆租赁源码1.7.2
易优cms汽车车辆租赁源码1.7.2

由于疫情等原因大家都开始习惯了通过互联网上租车服务的信息多方面,且获取方式简便,不管是婚庆用车、旅游租车、还是短租等租车业务。越来越多租车企业都开始主动把租车业务推向给潜在需求客户,所以如何设计一个租车网站,以便在同行中脱颖而出就重要了,易优cms针对租车行业市场需求、目标客户、盈利模式等,进行策划、设计、制作,建设一个符合用户与搜索引擎需求的租车网站源码。 网站首页

下载

立即学习go语言免费学习笔记(深入)”;

  • 按业务域拆流:不要所有服务都监听 event_stream,而是分 user_event_streampayment_event_stream
  • 避免「一个流 + 多个消费者组」做扇出:这会导致每组都收到全量消息,浪费带宽和解析开销;改用 XADD 时显式写入多个 stream(如同时发到 notify_streamaudit_stream
  • 幂等关键:每个消息必须带唯一 id(如 UUID 或业务单号 + 时间戳),消费者写入前先查 SETNX order_id_processed 1 EX 3600,失败则跳过
  • 别依赖 Redis 过期自动清理幂等键:高并发下可能误删,建议业务层主动维护生命周期或用 Lua 脚本原子判断+设置

连接与序列化容易被忽略的坑

Go 微服务常因连接复用不当或序列化不一致导致通信失败或 CPU 暴涨。

  • 连接池配置不生效:用 redis.NewClient 默认是单连接,必须显式设 PoolSize(如 10–30),并确保整个服务共用一个 *redis.Client 实例
  • 序列化陷阱:Go 结构体字段没加 json: tag,json.Marshal 后全是空对象;或用了 map[interface{}]interface{} 导致 Redis 存的是 "key":{} 这种无法反序列化的格式
  • 时间字段乱序:Redis Stream ID 基于毫秒时间戳 + 序列号,若服务系统时间不同步(如容器内没挂载 host 时间),可能导致消息 ID 乱序,XREADGROUP 拿不到预期数据
  • 错误信息典型:redis: nil reply from server → 通常是连接被服务端踢出(timeout 设置过短或 maxclients 超限);invalid character 'x' looking for beginning of value → 接收方用 json.Unmarshal 解析了非 JSON 字符串(比如原始字符串没包在 {} 里)

实际部署时,Streams 的 MAXLEN 策略(如 XADD ... MAXLEN ~ 10000)比单纯依赖 TTL 更可控;而真正难的是跨服务的错误传播——比如支付服务消费失败,怎么让订单服务知道要重试?这已经超出 Redis 能力范围,得靠补偿事务或 Saga 模式。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

177

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

226

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

336

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

208

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

388

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

194

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

189

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

191

2025.06.17

C++ 高性能计算与并行编程
C++ 高性能计算与并行编程

本专题专注于 C++ 在高性能计算(HPC)与并行编程中的应用,涵盖多线程、并发数据处理、OpenMP、MPI、GPU加速等技术。通过实际案例,帮助开发者掌握 如何利用 C++ 进行大规模数据计算和并行处理,提高程序的执行效率,适应高性能计算与数据密集型应用场景。

1

2026.01.08

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

Redis+MySQL数据库面试教程
Redis+MySQL数据库面试教程

共72课时 | 6.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号