缓存
mcache 是 Maltose 框架提供的一个高层次、通用的缓存组件。它通过适配器模式(Adapter Pattern)将底层具体的缓存实现(如内存、Redis)与上层缓存接口解耦,使得开发者可以用一套统一的 API 来操作不同的缓存介质。
特性
- 统一的 API: 无论是使用内存还是 Redis,缓存操作的 API 完全一致。
- 适配器模式: 支持通过适配器轻松切换或扩展缓存实现,默认提供高性能的内存适配器。
- LRU 淘汰策略: 默认的内存适配器支持基于 LRU (Least Recently Used) 策略的容量限制和自动淘汰。
- 丰富接口: 提供了
Get,Set,GetOrSet,SetIfNotExist,Remove等一系列丰富的缓存操作接口。 - 包级方法: 提供了包级别的便捷方法,底层使用一个默认的缓存实例,让简单使用场景的调用更加方便。
- 线程安全: 内置的内存适配器是线程安全的。
快速上手
mcache 提供了包级别的函数,可以直接使用,默认使用内存作为缓存。
package main
import (
"context"
"fmt"
"time"
"github.com/graingo/maltose/os/mcache"
)
func main() {
ctx := context.Background()
// 设置一个缓存,有效期 5 秒
err := mcache.Set(ctx, "my-key", "my-value", 5*time.Second)
if err != nil {
panic(err)
}
fmt.Println("Set cache: my-key = my-value")
// 获取缓存
val, _ := mcache.Get(ctx, "my-key")
fmt.Println("Get cache:", val.String()) // 输出: my-value
// 5秒后再次获取,此时缓存已过期
time.Sleep(6 * time.Second)
val, _ = mcache.Get(ctx, "my-key")
fmt.Println("Get after 6s:", val.IsNil()) // 输出: true
}实例与适配器
除了包级方法,您也可以创建自己的缓存实例,这在需要多个缓存或自定义缓存行为时非常有用。
使用内存缓存 (带 LRU)
您可以创建一个有容量限制的内存缓存。当缓存项数量超过容量时,最久未被使用的项将会被自动淘汰。
// 创建一个新的内存缓存实例,容量为 1000
cache := mcache.New(1000)
// 使用这个实例进行操作
cache.Set(ctx, "key1", "value1", 0) // 0 表示永不过期使用 Redis 缓存
Maltose 在 contrib/cache/redis 中提供了 Redis 适配器。您可以为缓存单独创建一个 mredis 实例,以获得更好的资源隔离。
import (
"context"
"fmt"
"time"
"github.com/graingo/maltose/contrib/cache/redis"
"github.com/graingo/maltose/database/mredis"
"github.com/graingo/maltose/os/mcache"
)
func main() {
ctx := context.Background()
// 1. 为缓存单独创建一个 Redis 客户端,使用独立的 db,避免与其他业务混淆
redisConfig := &mredis.Config{
Address: "127.0.0.1:6379",
Db: 1, // 推荐为缓存使用独立的 db
}
redisClient, err := mredis.New(redisConfig)
if err != nil {
panic(err)
}
// 2. 使用 redis 实例创建 Redis 缓存适配器
redisAdapter := redis.NewAdapterRedis(redisClient)
// 3. 使用适配器创建缓存实例
redisCache := mcache.NewWithAdapter(redisAdapter)
// 4. 后续使用 API 与内存缓存完全一致
redisCache.Set(ctx, "user:1", `{"name":"maltose"}`, time.Minute)
user, _ := redisCache.Get(ctx, "user:1")
fmt.Println(user.String())
}注意事项
关于 Clear/Size/Data/Keys/Values 等全局操作
与内存适配器不同,mcache 的 Redis 适配器本身没有数据分组的功能。如果多个 mcache 实例连接到同一个 Redis DB,它们将会共享这个 DB 内的所有数据。
因此,当您调用 Clear()、Size()、Data()、Keys()、Values() 这类方法时,它们操作的是整个 Redis DB,而不是像内存缓存那样只操作当前 mcache 实例的内部数据。
警告: 其中 Data()、Keys()、Values() 方法底层使用了 KEYS * 命令,这在生产环境中对大数据量的 Redis 会造成性能问题,甚至阻塞服务。强烈建议不要在生产环境中使用这些方法。Clear() 方法会清空当前 DB 的所有键值。这些都是与直觉可能相反的行为,请务必谨慎使用。
建议为缓存使用独立的 DB
鉴于上述操作的全局性,我们强烈建议您在使用 Redis 作为缓存时,通过配置为其指定一个独立的数据库(例如 db: 1, db: 2),而不是与其他业务数据(如会话、任务队列等)共用一个默认的 DB (db: 0)。这样可以有效隔离不同场景的数据,避免误操作带来的风险。
核心接口 Adapter
mcache 的灵活性来自于它的 Adapter 接口。任何实现了该接口的结构体,都可以作为 mcache 的底层驱动。
// Adapter 接口定义 (部分)
type Adapter interface {
Set(ctx context.Context, key string, value interface{}, duration time.Duration) error
Get(ctx context.Context, key string) (*mvar.Var, error)
Remove(ctx context.Context, keys ...string) (lastValue *mvar.Var, err error)
Close(ctx context.Context) error
// ... 其他方法
}这使得您可以非常轻松地实现自己的缓存适配器,例如对接 Memcached 或其他第三方缓存服务。
常用方法
除了基础的 Get、Set 方法,mcache 还提供了更丰富的缓存操作接口:
Contains - 检查键是否存在
exists, err := cache.Contains(ctx, "user:123")
if exists {
fmt.Println("缓存存在")
}Update - 更新现有键的值
更新缓存值,但不改变过期时间:
// 只更新值,保持原有的过期时间
err := cache.Update(ctx, "user:123", newUserData)UpdateExpire - 更新过期时间
只更新过期时间,不改变缓存值:
// 将过期时间延长到 1 小时
err := cache.UpdateExpire(ctx, "user:123", time.Hour)GetExpire - 获取剩余过期时间
// 获取缓存的剩余有效时间
ttl, err := cache.GetExpire(ctx, "user:123")
if err != nil {
// 处理错误
}
fmt.Printf("缓存还有 %v 过期\n", ttl)防止缓存击穿
缓存击穿是指热点数据过期时,大量并发请求同时访问数据库的问题。mcache 提供了专门的方法来处理这种场景。
SetIfNotExistFunc - 获取或设置
如果缓存不存在,则执行函数加载数据并设置缓存:
value, err := cache.SetIfNotExistFunc(ctx, "user:123",
func(ctx context.Context) (interface{}, error) {
// 从数据库加载数据
user, err := db.GetUser(ctx, 123)
if err != nil {
return nil, err
}
return user, nil
},
time.Hour, // 缓存 1 小时
)SetIfNotExistFuncLock - 加锁版本
这个方法会确保同一时刻只有一个 goroutine 执行加载函数,其他 goroutine 会等待结果:
value, err := cache.SetIfNotExistFuncLock(ctx, "user:123",
func(ctx context.Context) (interface{}, error) {
// 从数据库加载数据
return db.GetUser(ctx, 123)
},
time.Hour,
)工作原理:
- 第一个请求发现缓存不存在,获取锁并执行加载函数
- 后续并发请求会等待第一个请求完成
- 第一个请求完成后,所有请求都能获得结果
- 只有一次数据库查询,避免了缓存击穿
与 msync 的配合使用
对于更复杂的并发控制场景(如跨实例的请求合并、分布式环境下的缓存击穿),建议使用 msync 的 SingleFlight 组件:
import (
"github.com/graingo/maltose/os/mcache"
"github.com/graingo/maltose/util/msync"
)
sf := msync.NewSingleFlight()
// 使用 SingleFlight 包装缓存操作
value, err := sf.Do(ctx, "user:123", func(ctx context.Context) (interface{}, error) {
// 1. 先尝试从缓存获取
cached, err := cache.Get(ctx, "user:123")
if err == nil && !cached.IsNil() {
return cached.Interface(), nil
}
// 2. 缓存不存在,从数据库加载
user, err := db.GetUser(ctx, 123)
if err != nil {
return nil, err
}
// 3. 写入缓存
cache.Set(ctx, "user:123", user, time.Hour)
return user, nil
})msync.SingleFlight 的优势:
- 支持跨多个缓存实例的请求合并
- 提供更细粒度的控制(超时、错误共享等)
- 适用于分布式环境
- 可以获取请求是否是新鲜加载的(
DoEx方法)
更多并发控制能力,请参考 → 并发控制