Golang Web缓存核心是减少重复计算和数据库查询,优先用HTTP缓存头(Cache-Control、ETag)和内存/Redis缓存层;中小流量用go-cache(支持TTL),多实例用Redis(防穿透、雪崩),缓存策略须匹配数据更新频率与访问量。

用 Golang 实现 Web 应用缓存,核心是减少重复计算和数据库查询,把高频、低变动的数据暂存在内存或外部存储中。最常用且见效快的方式是结合 HTTP 缓存头 和 内存/Redis 缓存层,而不是一上来就堆复杂方案。
用 http.ServeMux + middleware 控制响应缓存
Go 标准库的 net/http 支持直接设置 Cache-Control、ETag、Last-Modified 等头部,让客户端或 CDN 自动缓存静态资源或稳定接口:
- 对不变内容(如 API 文档页、配置页),在 handler 中写:
w.Header().Set("Cache-Control", "public, max-age=3600") - 对可能更新的内容,用 ETag 做条件响应:生成内容哈希作为 ETag,收到
If-None-Match时比对,相同则返回 304 - 避免给登录态接口加强缓存——用户专属数据不能被共享缓存(如设为
private, max-age=60)
用 sync.Map 或 github.com/patrickmn/go-cache 做内存缓存
适合中小流量、数据量不大、重启可丢的场景(比如用户权限树、地区列表):
-
sync.Map是线程安全的,适合读多写少的键值缓存,但不支持自动过期,需自己定时清理或写 wrapper - 更推荐
github.com/patrickmn/go-cache:支持 TTL、自动驱逐、可选清理 goroutine,初始化简单:c := cache.New(5*time.Minute, 10*time.Minute) - 缓存 key 建议包含业务标识,如
"user:profile:" + userID,避免命名冲突
用 Redis 做分布式缓存(多实例共享)
当服务部署多个实例,或需要持久化、大容量、复杂数据结构(如排行榜、会话)时,Redis 是首选:
立即学习“go语言免费学习笔记(深入)”;
- 用
github.com/go-redis/redis/v9客户端,连接池配置好MinIdleConns和MaxConnAge - 读逻辑建议“先查缓存,未命中再查 DB,查到后回填缓存”,并注意缓存穿透(空结果也缓存短时间)、缓存雪崩(大量 key 同时过期)
- 写操作要同步失效缓存,例如更新用户信息后执行
DEL user:profile:123,而非更新缓存值(易不一致)
缓存策略要匹配业务真实需求
别为了缓存而缓存。几个关键判断点:
- 这个数据多久变一次?如果每分钟都改,缓存 1 秒都可能出问题
- 它被访问的频率高吗?QPS
- 查一次 DB 耗时多少?如果原生查询只要 2ms,加 Redis 反而变慢(网络 RTT + 序列化)
- 有没有缓存一致性要求?金融类数据慎用缓存,或必须搭配消息队列做最终一致
不复杂但容易忽略:上线前用 curl -I 检查响应头,用 redis-cli monitor 观察缓存读写,再压测对比 P95 延迟变化。










