JedisConnectionFactory中配置了哨兵后,会自动发现master节点,它是怎么作到的呢?redis
注:Spring-data-redis的版本: 1.8.7.RELEASE,Jedis的版本: 2.9.0.RELEASEspring
List-1 properties文件省略了apache
<bean class="redis.clients.jedis.JedisPoolConfig" id="jedisPoolConfig"> <property name="maxTotal" value="${redis.maxTotal}"/> <property name="maxIdle" value="${redis.maxIdle}"/> <property name="numTestsPerEvictionRun" value="${redis.numTestsPerEvictionRun}"/> <property name="timeBetweenEvictionRunsMillis" value="${redis.timeBetweenEvictionRunsMillis}"/> <property name="minEvictableIdleTimeMillis" value="${redis.minEvictableIdleTimeMillis}"/> <property name="softMinEvictableIdleTimeMillis" value="${redis.softMinEvictableIdleTimeMillis}"/> <property name="testOnReturn" value="${redis.testOnReturn}"/> <property name="maxWaitMillis" value="${redis.maxWaitMillis}"/> <property name="testOnBorrow" value="${redis.testOnBorrow}"/> </bean> <!-- redis集群配置 哨兵模式 --> <bean class="org.springframework.data.redis.connection.RedisSentinelConfiguration" id="sentinelConfiguration"> <property name="master"> <bean class="org.springframework.data.redis.connection.RedisNode"> <property name="name" value="${redis.master}"/> </bean> </property> <property name="sentinels"> <set> <bean class="org.springframework.data.redis.connection.RedisNode"> <constructor-arg name="host" value="${redis.sentinel.host1}"/> <constructor-arg name="port" value="${redis.sentinel.port1}"/> </bean> <bean class="org.springframework.data.redis.connection.RedisNode"> <constructor-arg name="host" value="${redis.sentinel.host2}"/> <constructor-arg name="port" value="${redis.sentinel.port2}"/> </bean> <bean class="org.springframework.data.redis.connection.RedisNode"> <constructor-arg name="host" value="${redis.sentinel.host3}"/> <constructor-arg name="port" value="${redis.sentinel.port3}"/> </bean> </set> </property> </bean> <bean id="redisConnectionFactory" class="org.springframework.data.redis.connection.jedis.JedisConnectionFactory"> <constructor-arg name="sentinelConfig" ref="sentinelConfiguration"/> <constructor-arg name="poolConfig" ref="jedisPoolConfig"/> <property name="database" value="${redis.dbIndex}"/> </bean>
这样咱们就在JedisConnectionFactory中配置了pool和哨兵信息。spa
看以下图1所示:线程
图1 JedisConnectionFactory的继承图code
若是了解过SpringIOC的,那么看到InitializingBean,应该知道JedisConnectionFactory为何继承它了。看以下图2所示xml
图2 JedisConnectionFactory的afterPropertiesSet时序图blog
重点在步骤12中,MasterListener继承了Thread,它是个单独的线程,经过Jedis监听Redis的哨兵,若是收到master改变的消息,那么会修改JedisFactory。以下图3所示是MasterListener收到master节点变化的消息: 继承
图3 MasterListener收到master改变的消息it
图3中,message的内容表示,以前6482端口的redis是master,如今6483端口的才是master。
收到master改变的消息后,会修改JedisFactory,进而会影响connection的获取。值得注意的是,链接池底层上使用的是apache的common-pool的org.apache.commons.pool2.impl.GenericObjectPool,有兴趣的读者能够去看下common-pool的实现,我在这里就再也不深刻common-pool的实现。提示: dbcp链接池使用的也是common-pool,若是你理解了common-pool的实现,那么会明白链接池的不少参数是什么意思(好比maxActive、maxIdle等)。