go-zero/doc/sharedcalls.md
masonchen2014 2896ef1a49
Sharedcalls.md (#69)
* sharedcalls.md

* markdownlint

Co-authored-by: chenmusheng <chenmusheng@laoyuegou.com>
2020-09-14 10:13:33 +08:00

6.9 KiB
Raw Blame History

进程内共享调用SharedCalls

go-zero微服务框架中提供了许多开箱即用的工具好的工具不仅能提升服务的性能而且还能提升代码的鲁棒性避免出错实现代码风格的统一方便他人阅读等等。

本文主要讲述进程内共享调用神器SharedCalls

使用场景

并发场景下可能会有多个线程协程同时请求同一份资源如果每个请求都要走一遍资源的请求过程除了比较低效之外还会对资源服务造成并发的压力。举一个具体例子比如缓存失效多个请求同时到达某服务请求某资源该资源在缓存中已经失效此时这些请求会继续访问DB做查询会引起数据库压力瞬间增大。而使用SharedCalls可以使得同时多个请求只需要发起一次拿结果的调用其他请求"坐享其成",这种设计有效减少了资源服务的并发压力,可以有效防止缓存击穿。

例如高并发情况下多个请求同时查询某用户信息以下是利用SharedCalls工具包实现的用户信息获取方法注意⚠️go-zero框架中已经提供了针对此场景的代码实现以下代码只为说明SharedCalls的使用方式具体可参看sqlcmongoc等处代码)

// 用户信息结构体
type UserInfo struct {
    UserId int
    Name   string
    Age    int
}

// 用户信息model对象
type UserModel struct {
    db      *sql.DB
    rds     *redis.Client
    barrier syncx.SharedCalls
}

// 利用SharedCalls实现的用户信息获取方法
func (um *UserModel) GetUserInfoEffectively(userId int) (*UserInfo, error) {

    // SharedCalls.Do方法提供两个参数第一个参数是资源获取期间的唯一标识
    // 第二个参数是指定一个fun()(interface{},error) 方法来真正获取资源
    val, err := um.barrier.Do(fmt.Sprintf("uId:{%d}:applying", userId), func() (interface{}, error) {
        userInfo := &UserInfo{}

        // 从redis中获取用户信息
        userInfoStr, err := um.rds.Get(fmt.Sprintf("uId:{%d}", userId)).Result()
        if err != nil && err != redis.Nil {
            return nil, err
        }

        // 缓存不为空,解析缓存中的用户数据
        if len(userInfoStr) > 0 {
            if err := json.Unmarshal(([]byte)(userInfoStr),userInfo);err!=nil {
                return nil, err
            }
        } else {

            // 缓存为空从db中获取用户数据
            if err := um.db.QueryRow("select id,name,age from users where id=?", userId).Scan(&userInfo.UserId,&userInfo.Name, &userInfo.Age); err != nil {
                return nil, err
            }

            // 将用户信息保存到缓存
            userInfoBytes, err := json.Marshal(userInfo)
            if err == nil {
                um.rds.Set(fmt.Sprintf("uId:{%d}", userId), userInfoBytes, 5*time.Second)
            }
        }

        return userInfo, nil
    })

    // 判断获取用户过程中是否发生错误
    if err != nil {
        return nil, err
    }

    // 返回用户信息
    return val.(*UserInfo), nil
}

关键源码分析

  • SharedCalls interface提供了Do和DoEx两种方法的抽象

    // SharedCalls接口提供了Do和DoEx两种方法
    type SharedCalls interface {
     Do(key string, fn func() (interface{}, error)) (interface{}, error)
     DoEx(key string, fn func() (interface{}, error)) (interface{}, bool, error)
    }
    
  • SharedCalls interface的具体实现sharedGroup

    // call代表对指定资源的一次请求
    type call struct {
     wg  sync.WaitGroup  // 用于协调各个请求goroutine之间的资源共享
     val interface{}     // 用于保存请求的返回值
     err error           // 用于保存请求过程中发生的错误
    }
    
    type sharedGroup struct {
     calls map[string]*call
     lock  sync.Mutex
    }
    
  • sharedGroup的Do方法

    • key参数可以理解为资源的唯一标识。
    • fn参数真正获取资源的方法。
    • 处理过程分析:
    // 当多个请求同时使用Do方法请求资源时
    func (g *sharedGroup) Do(key string, fn func() (interface{}, error)) (interface{}, error) {
    
      // 先申请加锁
      g.lock.Lock()
    
      // 根据key获取对应的call结果,并用变量c保存
      if c, ok := g.calls[key]; ok {
    
        // 拿到call以后释放锁此处call可能还没有实际数据只是一个空的内存占位
        g.lock.Unlock()
    
        // 调用wg.Wait判断是否有其他goroutine正在申请资源如果阻塞说明有其他goroutine正在获取资源
        c.wg.Wait()
    
        // 当wg.Wait不再阻塞表示资源获取已经结束可以直接返回结果
        return c.val, c.err
    
      }
    
    
      // 没有拿到结果则调用makeCall方法去获取资源注意此处仍然是锁住的可以保证只有一个goroutine可以调用makecall
      c := g.makeCall(key, fn)
    
      // 返回调用结果
      return c.val, c.err
    
    }
    
  • sharedGroup的DoEx方法

    • 和Do方法类似只是返回值中增加了布尔值表示值是调用makeCall方法直接获取的还是取的共享成果
    func (g *sharedGroup) DoEx(key string, fn func() (interface{}, error)) (val interface{}, fresh bool, err error) {
     g.lock.Lock()
     if c, ok := g.calls[key]; ok {
      g.lock.Unlock()
      c.wg.Wait()
      return c.val, false, c.err
     }
    
     c := g.makeCall(key, fn)
     return c.val, true, c.err
    }
    
  • sharedGroup的makeCall方法

    • 该方法由Do和DoEx方法调用是真正发起资源请求的方法。
    //进入makeCall的一定只有一个goroutine因为要拿锁锁住的
    func (g *sharedGroup) makeCall(key string, fn func() (interface{}, error)) *call {
    
      // 创建call结构用于保存本次请求的结果
      c := new(call)
    
      // wg加1用于通知其他请求资源的goroutine等待本次资源获取的结束
      c.wg.Add(1)
    
      // 将用于保存结果的call放入map中以供其他goroutine获取
      g.calls[key] = c
    
      // 释放锁这样其他请求的goroutine才能获取call的内存占位
      g.lock.Unlock()
    
    
      defer func() {
    
        // delete key first, done later. can't reverse the order, because if reverse,
        // another Do call might wg.Wait() without get notified with wg.Done()
        g.lock.Lock()
        delete(g.calls, key)
        g.lock.Unlock()
    
        // 调用wg.Done通知其他goroutine可以返回结果这样本批次所有请求完成结果的共享
        c.wg.Done()
    
      }()
    
    
      // 调用fn方法将结果填入变量c中
      c.val, c.err = fn()
      return c
    
    }
    

最后

本文主要介绍了go-zero框架中的 SharedCalls工具对其应用场景和关键代码做了简单的梳理希望本篇文章能给大家带来一些收获。