go-zero缓存设计之持久层缓存

缓存设计原理

我们对缓存是只删除,不做更新,一旦DB里数据出现修改,我们就会直接删除对应的缓存,而不是去更新。

我们看看删除缓存的顺序怎样才是正确的。

  • 先删除缓存,再更新DB

redis-cache-01

我们看两个并发请求的情况,A请求需要更新数据,先删除了缓存,然后B请求来读取数据,此时缓存没有数据,就会从DB加载数据并写回缓存,然后A更新了DB,那么此时缓存内的数据就会一直是脏数据,直到缓存过期或者有新的更新数据的请求。如图

redis-cache-02

  • 先更新DB,再删除缓存

    redis-cache-03

A请求先更新DB,然后B请求来读取数据,此时返回的是老数据,此时可以认为是A请求还没更新完,最终一致性,可以接受,然后A删除了缓存,后续请求都会拿到最新数据,如图 redis-cache-04

让我们再来看一下正常的请求流程:

  • 第一个请求更新DB,并删除了缓存
  • 第二个请求读取缓存,没有数据,就从DB读取数据,并回写到缓存里
  • 后续读请求都可以直接从缓存读取 redis-cache-05

我们再看一下DB查询有哪些情况,假设行记录里有ABCDEFG七列数据:

  • 只查询部分列数据的请求,比如请求其中的ABC,CDE或者EFG等,如图 redis-cache-06

  • 查询单条完整行记录,如图 redis-cache-07

  • 查询多条行记录的部分或全部列,如图 redis-cache-08

对于上面三种情况,首先,我们不用部分查询,因为部分查询没法缓存,一旦缓存了,数据有更新,没法定位到有哪些数据需要删除;其次,对于多行的查询,根据实际场景和需要,我们会在业务层建立对应的从查询条件到主键的映射;而对于单行完整记录的查询,go-zero 内置了完整的缓存管理方式。所以核心原则是:go-zero 缓存的一定是完整的行记录

下面我们来详细介绍 go-zero 内置的三种场景的缓存处理方式:

  • 基于主键的缓存
    1. PRIMARY KEY (`id`)

这种相对来讲是最容易处理的缓存,只需要在 redis 里用 primary key 作为 key 来缓存行记录即可。

  • 基于唯一索引的缓存 redis-cache-09

在做基于索引的缓存设计的时候我借鉴了 database 索引的设计方法,在 database 设计里,如果通过索引去查数据时,引擎会先在 索引->主键 的 tree 里面查找到主键,然后再通过主键去查询行记录,就是引入了一个间接层去解决索引到行记录的对应问题。在 go-zero 的缓存设计里也是同样的原理。

基于索引的缓存又分为单列唯一索引和多列唯一索引:

但是对于 go-zero 来说,单列和多列只是生成缓存 key 的方式不同而已,背后的控制逻辑是一样的。然后 go-zero 内置的缓存管理就比较好的控制了数据一致性问题,同时也内置防止了缓存的击穿、穿透、雪崩问题(这些在 gopherchina 大会上分享的时候仔细讲过,见后续 gopherchina 分享视频)。

另外,go-zero 内置了缓存访问量、访问命中率统计,如下所示:

  1. dbcache(sqlc) - qpm: 5057, hit_ratio: 99.7%, hit: 5044, miss: 13, db_fails: 0

可以看到比较详细的统计信息,便于我们来分析缓存的使用情况,对于缓存命中率极低或者请求量极小的情况,我们就可以去掉缓存了,这样也可以降低成本。

  • 单列唯一索引如下:

    1. UNIQUE KEY `product_idx` (`product`)
  • 多列唯一索引如下:

    1. UNIQUE KEY `vendor_product_idx` (`vendor`, `product`)

    缓存代码解读

1.基于主键的缓存逻辑

redis-cache-10

具体实现代码如下:

  1. func (cc CachedConn) QueryRow(v interface{}, key string, query QueryFn) error {
  2. return cc.cache.Take(v, key, func(v interface{}) error {
  3. return query(cc.db, v)
  4. })
  5. }

这里的 Take 方法是先从缓存里去通过 key 拿数据,如果拿到就直接返回,如果拿不到,那么就通过 query 方法去 DB 读取完整行记录并写回缓存,然后再返回数据。整个逻辑还是比较简单易懂的。

我们详细看看 Take 的实现:

  1. func (c cacheNode) Take(v interface{}, key string, query func(v interface{}) error) error {
  2. return c.doTake(v, key, query, func(v interface{}) error {
  3. return c.SetCache(key, v)
  4. })
  5. }

Take 的逻辑如下:

  • 用 key 从缓存里查找数据
  • 如果找到,则返回数据
  • 如果找不到,用 query 方法去读取数据
  • 读到后调用 c.SetCache(key, v) 设置缓存

其中的 doTake 代码和解释如下:

  1. // v - 需要读取的数据对象
  2. // key - 缓存key
  3. // query - 用来从DB读取完整数据的方法
  4. // cacheVal - 用来写缓存的方法
  5. func (c cacheNode) doTake(v interface{}, key string, query func(v interface{}) error,
  6. cacheVal func(v interface{}) error) error {
  7. // 用barrier来防止缓存击穿,确保一个进程内只有一个请求去加载key对应的数据
  8. val, fresh, err := c.barrier.DoEx(key, func() (interface{}, error) {
  9. // 从cache里读取数据
  10. if err := c.doGetCache(key, v); err != nil {
  11. // 如果是预先放进来的placeholder(用来防止缓存穿透)的,那么就返回预设的errNotFound
  12. // 如果是未知错误,那么就直接返回,因为我们不能放弃缓存出错而直接把所有请求去请求DB,
  13. // 这样在高并发的场景下会把DB打挂掉的
  14. if err == errPlaceholder {
  15. return nil, c.errNotFound
  16. } else if err != c.errNotFound {
  17. // why we just return the error instead of query from db,
  18. // because we don't allow the disaster pass to the DBs.
  19. // fail fast, in case we bring down the dbs.
  20. return nil, err
  21. }
  22. // 请求DB
  23. // 如果返回的error是errNotFound,那么我们就需要在缓存里设置placeholder,防止缓存穿透
  24. if err = query(v); err == c.errNotFound {
  25. if err = c.setCacheWithNotFound(key); err != nil {
  26. logx.Error(err)
  27. }
  28. return nil, c.errNotFound
  29. } else if err != nil {
  30. // 统计DB失败
  31. c.stat.IncrementDbFails()
  32. return nil, err
  33. }
  34. // 把数据写入缓存
  35. if err = cacheVal(v); err != nil {
  36. logx.Error(err)
  37. }
  38. }
  39. // 返回json序列化的数据
  40. return jsonx.Marshal(v)
  41. })
  42. if err != nil {
  43. return err
  44. }
  45. if fresh {
  46. return nil
  47. }
  48. // got the result from previous ongoing query
  49. c.stat.IncrementTotal()
  50. c.stat.IncrementHit()
  51. // 把数据写入到传入的v对象里
  52. return jsonx.Unmarshal(val.([]byte), v)
  53. }

2. 基于唯一索引的缓存逻辑

因为这块比较复杂,所以我用不同颜色标识出来了响应的代码块和逻辑,block 2 其实跟基于主键的缓存是一样的,这里主要讲 block 1 的逻辑。 redis-cache-11

代码块的 block 1 部分分为两种情况:

  • 通过索引能够从缓存里找到主键,此时就直接用主键走 block 2 的逻辑了,后续同上面基于主键的缓存逻辑

  • 通过索引无法从缓存里找到主键

    • 通过索引从DB里查询完整行记录,如有 error,返回
    • 查到完整行记录后,会把主键到完整行记录的缓存和索引到主键的缓存同时写到 redis
    • 返回所需的行记录数据
  1. // v - 需要读取的数据对象
  2. // key - 通过索引生成的缓存key
  3. // keyer - 用主键生成基于主键缓存的key的方法
  4. // indexQuery - 用索引从DB读取完整数据的方法,需要返回主键
  5. // primaryQuery - 用主键从DB获取完整数据的方法
  6. func (cc CachedConn) QueryRowIndex(v interface{}, key string, keyer func(primary interface{}) string,
  7. indexQuery IndexQueryFn, primaryQuery PrimaryQueryFn) error {
  8. var primaryKey interface{}
  9. var found bool
  10. // 先通过索引查询缓存,看是否有索引到主键的缓存
  11. if err := cc.cache.TakeWithExpire(&primaryKey, key, func(val interface{}, expire time.Duration) (err error) {
  12. // 如果没有索引到主键的缓存,那么就通过索引查询完整数据
  13. primaryKey, err = indexQuery(cc.db, v)
  14. if err != nil {
  15. return
  16. }
  17. // 通过索引查询到了完整数据,设置found,后面直接使用,不需要再从缓存读取数据了
  18. found = true
  19. // 将主键到完整数据的映射保存到缓存里,TakeWithExpire方法已经将索引到主键的映射保存到缓存了
  20. return cc.cache.SetCacheWithExpire(keyer(primaryKey), v, expire+cacheSafeGapBetweenIndexAndPrimary)
  21. }); err != nil {
  22. return err
  23. }
  24. // 已经通过索引找到了数据,直接返回即可
  25. if found {
  26. return nil
  27. }
  28. // 通过主键从缓存读取数据,如果缓存没有,通过primaryQuery方法从DB读取并回写缓存再返回数据
  29. return cc.cache.Take(v, keyer(primaryKey), func(v interface{}) error {
  30. return primaryQuery(cc.db, v, primaryKey)
  31. })
  32. }

我们来看一个实际的例子

  1. func (m *defaultUserModel) FindOneByUser(user string) (*User, error) {
  2. var resp User
  3. // 生成基于索引的key
  4. indexKey := fmt.Sprintf("%s%v", cacheUserPrefix, user)
  5. err := m.QueryRowIndex(&resp, indexKey,
  6. // 基于主键生成完整数据缓存的key
  7. func(primary interface{}) string {
  8. return fmt.Sprintf("user#%v", primary)
  9. },
  10. // 基于索引的DB查询方法
  11. func(conn sqlx.SqlConn, v interface{}) (i interface{}, e error) {
  12. query := fmt.Sprintf("select %s from %s where user = ? limit 1", userRows, m.table)
  13. if err := conn.QueryRow(&resp, query, user); err != nil {
  14. return nil, err
  15. }
  16. return resp.Id, nil
  17. },
  18. // 基于主键的DB查询方法
  19. func(conn sqlx.SqlConn, v, primary interface{}) error {
  20. query := fmt.Sprintf("select %s from %s where id = ?", userRows, m.table)
  21. return conn.QueryRow(&resp, query, primary)
  22. })
  23. // 错误处理,需要判断是否返回的是sqlc.ErrNotFound,如果是,我们用本package定义的ErrNotFound返回
  24. // 避免使用者感知到有没有使用缓存,同时也是对底层依赖的隔离
  25. switch err {
  26. case nil:
  27. return &resp, nil
  28. case sqlc.ErrNotFound:
  29. return nil, ErrNotFound
  30. default:
  31. return nil, err
  32. }
  33. }

所有上面这些缓存的自动管理代码都是可以通过 goctl 自动生成的,我们团队内部 CRUD 和缓存基本都是通过 goctl 自动生成的,可以节省大量开发时间,并且缓存代码本身也是非常容易出错的,即使有很好的代码经验,也很难每次完全写对,所以我们推荐尽可能使用自动的缓存代码生成工具去避免错误。

猜你想看