今天要跟大家剖析的是redis.v3,一个在Golang上的redis客户端库。

redis本身的内容,我就不多说了,有很多书籍可以查阅。

比方说: 《Redis实战-译本》《Redis设计与实现-第一版》《Redis设计与实现-第二版》,这三本书都是跟黄健宏有关。

如果想深入了解和学习Redis,那以上两本书(三本书)都值得拥有。

今天我不是来推荐书的,所以不再多说,直接来干货部分。

redis.v3

重点介绍以下几个文件redis.go,command.go,commands.go,pool.go

redis初始化:

redisClient := redis.NewClient(redis.Options{})

在NewClient中重要的是Options,在源代码redis.go中的L92有定义。

初始化之后,就可以开始使用redisClient了。

举例说明,SortedSet(有序集合)

我今天要说的是,在一系列设置之后,我们允许会有以下的log输出。

ZADD one: 0 [redis.go L52]

跟踪源码,我们发现L52的内容是c.opt.Logger(cmd.String())

就这一行就隐含着多个信息。

  1. redisClient的Optins配置的Logger
  2. redisClient cmd字符串表达

通过源码,可以很明显的知道cmd是Cmder类型。

type Cmder interface {
args() []string
parseReply(*bufio.Reader) error
setErr(error)
reset()

writeTimeout() *time.Duration
readTimeout() *time.Duration
clusterKey() string

Err() error
String() string

}

所以直接看String的方法实现是什么,就能得知,log中输出的内容是从何而来。

return cmdString(cmd, cmd.val)

cmdString的实现

s := strings.Join(cmd.args(), " ")
if err := cmd.Err(); err != nil {
    return s + ": " + err.Error()
}
if val != nil {
    return s + ": " + fmt.Sprint(val)
}
return s

看到这里,我相信大家应该知道 ZADD one: 0 的来源了吧。

查看commands.go中ZADD函数,

args := make([]string, 2+2*len(members))
args[0] = "ZADD"
args[1] = key
for i, m := range members {
    args[2+2*i] = formatFloat(m.Score)
    args[2+2*i+1] = m.Member
}
cmd := NewIntCmd(args...)
c.Process(cmd)

这段代码,需要特别关注cmd := NewIntCmd(args...), 它的核心方法parseReply,解析之后,我们可以看到会把结果值赋值给cmd.val,疑惑出现了,刚刚打印的为什么是0呢?

我们再回过头来看redis.go中的process方法L42,L52,

cmd.reset()紧接着就是cmd.String(),那串打印,大家是否明白了呢!?

还不明白么?

请看command.goL246,L247,cmd.val=0

追根溯源,终于知道那一串打印是从何而来了,对redis.v3的客户端内部实现也更熟悉了,其他cmd都类似,还想了解更多就仔细阅读源代码吧。

不知道算不算是干货呢,反正我明白了其中的原因,希望能对大家有所帮助。


茶歇驿站

一个可以让你停下来看一看,在茶歇之余给你帮助的小站。

这里的内容主要是后端技术,个人管理,团队管理,以及其他个人杂想。