redis集群启动流程关键类(源码版)
以Spring集成redis为例来说明,版本是2.8.2。
一,RedisProperties
解析配置文件中的配置项,这个class上加了一个@ConfigurationProperties注解,指定了prefix为spring.redis,也就是说处理配置文件中前缀为spring.redis的配置项。
RedisProperties定义的属性如下:
int database = 0
String url
String host = "localhost"
String password
int port = 6379
boolean ssl
int timeout
RedisProperties.Pool:封装了maxIdle、minIdle、maxActive、maxWait这4个配置项。
RedisProperties.Sentinel:封装了master、nodes配置项。
RedisProperties.Cluster:封装了nodes、maxRedirects配置项。
二,Jedis
jedis包下提供的一个类,封装了redis的常用命令,1700多行代码,基本的redis操作都封装了。
三,JedisConnection
JedisConnection继承了抽象类
AbstractRedisConnection,AbstractRedisConnection并没有提供什么额外功能。AbstractRedisConnection实现了RedisConnection接口,这个接口定义了8个方法:
// 关闭连接
void close()
// 查看连接是否关闭
boolean isClosed();
// 获取JedisCluster
Object getNativeConnection();
// 判断是否是队列,相对于redis管道来说的
boolean isQueueing();
// 判断是否支持管道命令
boolean isPipeline();
// 获取管道命令执行结果
List<Object> closePioeline();
// 获取哨兵连接,jedis2.8.2不支持哨兵连接
RedisSentinelConnection getSentinelConnection();
然后RedisConnection接口实现了RedisCommands接口,RedisCommands接口只有一个方法,那就是执行redis命令。
综合来看,JedisConnection的功能其实就是:
1,建立和关闭与redis集群的连接。
2,判断如何执行redis命令,管道方式执行还是非管道方式执行。
3,执行redis命令。
四,RedisOperations接口和RedisTemplate
封装了对redis的5种数据类型的操作。这个接口的实现类就是大名鼎鼎的RedisTemplate,我们操作redis数据的入口。
五,RedisAutoConfiguration
RedisAutoConfiguration包含2个内部类:
1,RedisConfiguration
提供了创建redisTemplate的方法。
2,RedisConnectionConfiguration
提供了配置redis连接和创建redis连接的一些方法。
六,RedisConnectionFactory
RedisConnectionFactory接口的实现类有2个:
JedisConnectionFactory
LettuceConnectionFactory
Spring boot2.x默认使用LettuceConnectionFactory,LettuceConnectionFactory支持redis的管道特性,但是JedisConnectionFactory默认是不支持的。
JedisConnectionFactory有下面这段代码:
return poolConfig != null ? new JedisCluster(...) : new JedisCluster(...);
括号中的参数这里省略了,我对比了几遍,一模一样。这应该是Spring开发者留下的一个bug了。这个spring-data-redis的版本是1.7.8.RELEASE。
七,JedisPoolConfig
redis连接池配置,设置redis连接池的相关参数,例如:
maxTotal:最大连接数,默认值8。
maxIdle:最大闲置连接数,默认值8。
minIdle:最小闲置连接数,默认值0。
八,RedisSentinelConfiguration
RedisSentinelConfiguration主要的功能就是:redis哨兵节点配置项解析。解析完成以后会放到Set<RedisNode>集合中。
九,RedisClusterConfiguration
RedisClusterConfiguration主要的功能就是:redis集群节点配置项解析。解析完成以后会放到Set<RedisNode>集合中。
十,JedisCluster
定义集群的各种原子操作,比如hset(),所有的这些集群原子操作都是经过了JedisClusterCommand的包装,也就是说借助JedisClusterCommand来实现的。
接下来我们有必要来了解一下这个JedisClusterCommand。
JedisClusterCommand主要提供了4个方法:
1,run(String key):执行单个类型为String的key的操作。
2,run(int keyCount, String… keys):执行多个类型为String的key的操作。
3,runBinary(byte[] key):执行单个类型为byte数组的key的操作。
4,runBinary(int keyCount, byte[] … keys):执行多个类型为byte数组的key的操作。
我们看run(String key)的源码:
return this.runWithRetries(key, this.maxAttempts, false, false);
this.runWithRetries方法的四个入参如下:
key:键
this.maxAttempts:最大重试次数,这个参数必须大于0。
tryRandomNode:重试策略,随机连接redis集群中的节点。具体逻辑是:获取redis集群缓存JedisClusterInfoCache,然后从缓存拿到redis连接池JedisPool,遍历JedisPool,获取一个Jedis实例,ping一下,ping通了,就直接返回。
asking:重试策略,asking重试策略很特殊,只有当重定向的槽处于迁移状态时,才会使用asking重定向。与asking相对应的,还有一个moved重定向,指的是当前访问的key迁移到了其他槽,这里是迁移完成状态。
OK,搞清楚了这几个参数的含义,这个方法就好理解了。
this.runWithRetries首先会判断是否是asking重试策略,如果是,则执行下面的逻辑连接集群的某个节点:
connection = (Jedis)this.askConnection.get();
connection.asking();
asking = false;
这里需要注意,this.askConnection是ThreadLocal<Jedis>。
然后判断是否是随机重试策略,如果是,则执行下面的逻辑连接集群的某个节点:
connection = this.connectionHandler.getConnection();
如果既不是重试策略,又不是随机重试策略,则说明有可能是第一次访问redis节点,也有可能是重试访问redis节点,总之,使用
JedisClusterCRC16哈希算法计算key的hash值,找到对应的槽,然后连接,源码如下:
connection = this.connectionHandler.getConnectionFromSlot(JedisClusterCRC16.getSlot(key));
连接上redis节点以后,根据key执行对于的操作,然后返回:
return this.execute(connection);
这里的this.execute方法是JedisClusterCommand定于的一个抽象方法,具体实现在JedisCluster中。最终是调用了Jedis中定于的各种redis命令的实现。
好了,什么这些分析就是JedisClusterCommand的具体实现,说白了,JedisClusterCommand其实就是一个模板,redis集群执行各种redis命令实现的操作模板。
十一,JedisClusterCRC16
JedisClusterCRC16是Spring对redis哈希算法CRC16的实现。感兴趣的朋友可以看看。