楔子:某个时间,由于不清不楚的某些原因,导致了一次严重的线上事故。后来,开发不清不楚的配合把项目升级到了 Redis 高可用集群的哨兵模式(Redis-Sentinel),再后来,我们逐渐的又不清不楚的淡忘了这件事。节点化的工作很容易导致一定程度上只知其然而不知其所以然,这是项目开发中的一个众相。回想起来,我还是想记点什么。
该篇可以为 Redis 容灾+高可用 应用场景提供解决方案。
本文为 上一击 的代码验证部分,是以 Redis进击(三)搭建Redis高可用集群的哨兵模式(Redis-Sentinel)【Windows环境】为基础进行的测试和验证。所以在进行测试或验证前,请确保以下前置任务正常:
注:如果主节点/从节点的服务端、Sentinel 没有全部正常开启的话,会出现异常:JedisConnectionException: All sentinels down, cannot determine where is 127.0.0.1 master is running...
场景一:Spring 项目中配置和使用 Redis 高可用集群的哨兵模式(Redis-Sentinel)
redis.clients
jedis
2.9.0
(1)classpath:redis.properties
redis.host=127.0.0.1
redis.port=6380
redis.timeout=3000
redis.pool.maxIdle=50
redis.pool.testOnBorrow=true
redis.pool.maxWaitMillis=3000
redis.sentinel.master=mymaster
redis.sentinel.node1=127.0.0.1:26379
redis.sentinel.node2=127.0.0.1:26380
redis.sentinel.node3=127.0.0.1:26381
注意:redis.sentinel.master 配置项的值必须与主节点/主服务器 sentinel.conf 配置文件中的 sentinel monitor mymaster 127.0.0.1 6379 3 的 主节点/主服务器名称 mymaster 保持一致。
否则,会出现异常:JedisException: Can connect to sentinel, but 127.0.0.1:6380 seems to be not monitored...
(2)classpath:spring-redis.xml
${redis.sentinel.node1}
${redis.sentinel.node2}
${redis.sentinel.node3}
配置 testApplicationContext.xml
classpath:redis/redis.properties
测试类:
package com.mwei.remote.redis;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisSentinelPool;
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:testApplicationContext.xml")
public class JedisPoolTest {
@Autowired
private JedisSentinelPool jedisPool;
@Test
public void testJedisPool4Get() {
Jedis jedis = null;
try {
jedis = jedisPool.getResource();
jedis.set("uid_010_name", "美味物语");
System.out.println("uid_010_name : " + jedis.get("uid_010_name"));
} finally {
if (jedis != null) {
jedis.close();
}
}
}
}
输出结果,验证OK:
uid_010_name : 美味物语
与 JUnit 测试方式不一样的是,下面的测试方法省去了诸多配置项,只需要进行测试类编写和运行:
package com.weix.service.redis;
import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPoolConfig;
import redis.clients.jedis.JedisSentinelPool;
import java.util.HashSet;
import java.util.Set;
public class JedisUtil {
private static JedisSentinelPool jedisPool = null;
// 自带的哨兵模式 JedisSentinelPool, 并在一开始初始化连接池
static {
try {
JedisPoolConfig config = new JedisPoolConfig();
// 资源池中的最大连接数
// 如果赋值为-1,则表示不限制;如果pool已经分配了maxActive个jedis实例,则此时pool的状态为exhausted(耗尽)
config.setMaxTotal(Integer.valueOf(50));
// 控制一个pool最多有多少个状态为idle(空闲的)的jedis实例
config.setMaxIdle(Integer.valueOf(50));
// 表示当borrow(引入)一个jedis实例时,最大的等待时间,如果超过等待时间,则直接抛出JedisConnectionException
config.setMinEvictableIdleTimeMillis(Integer.valueOf(-1));
// 在borrow一个jedis实例时,是否提前进行validate操作;如果为true,则得到的jedis实例均是可用的
config.setTestOnBorrow(Boolean.valueOf(true));
// Setinel哨兵群
// Setinel客户端提供了master自动发现功能
Set sentinels = new HashSet<>();
sentinels.add("127.0.0.1:26379");
sentinels.add("127.0.0.1:26380");
sentinels.add("127.0.0.1:26381");
// master名称必须要和配置文件sentinel.conf中配置保持一致
jedisPool = new JedisSentinelPool("mymaster", sentinels, config);
} catch (Exception e) {
e.printStackTrace();
}
}
/**
* 构建redis连接池
*
* @return JedisPool
*/
public static JedisSentinelPool getJedisPool() {
return jedisPool;
}
/**
* 测试redis线程池是否正常
*
* @param args
*/
public static void main(String[] args) {
JedisSentinelPool jedisSentinelPool = JedisUtil.getJedisPool();
Jedis jedis = jedisSentinelPool.getResource();
System.out.println("jedis = " + jedis);
jedis.set("uid_020_name", "美味进击");
System.out.println("uid_020_name : " + jedis.get("uid_020_name"));
// Jedis 2.9.0 版本及以上的 JedisPool 的 returnBrokenResource() 和 returnResource() 方法被标注废弃了,取而代之的是 Jedis 的 close()
if (jedis != null) {
jedis.close();
}
}
}
输出结果,验证OK:
jedis = redis.clients.jedis.Jedis@379619aa
uid_020_name : 美味进击
场景二:Spring 项目中配置和使用 Redis 高可用集群的哨兵模式(Redis-Sentinel)
org.springframework.boot
spring-boot-starter-data-redis
2.0.4.RELEASE
(1)classpath:application.yml
spring:
redis:
jedis:
pool:
max-idle: 8
max-wait: 3000
sentinel:
master: mymaster
nodes:
- 127.0.0.1:26379
- 127.0.0.1:26380
- 127.0.0.1:26381
启动类:
package com.mwei;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
/**
* 注解@SpringBootApplication是Sprnig Boot项目的核心注解,主要目的是开启自动配置
*/
@SpringBootApplication
//@PropertySource("file:/etc/dbprivatekey.properties") // Redis配置。因为没有配置也没有用到密码,所以这里先注释掉
public class MWeiSpringBootApplication {
/**
* 标准的Java应用的main的方法,主要作用是作为项目启动的入口
* @param args
*/
public static void main(String[] args) {
SpringApplication.run(MWeiSpringBootApplication.class, args);
}
}
测试类:
package com.mwei.utils;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.boot.web.servlet.support.SpringBootServletInitializer;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.test.context.junit4.SpringRunner;
@RunWith(SpringRunner.class)
@SpringBootTest
public class JedisUtilTest extends SpringBootServletInitializer {
@Autowired
private RedisTemplate redisTemplate;
@Test
public void testRedisTemplate4Get() {
this.redisTemplate.opsForValue().set("uid_030_name", "美味书签");
System.out.println("uid_030_name : " + this.redisTemplate.opsForValue().get("uid_030_name"));
}
}
输出结果,验证OK:
uid_030_name : 美味书签
好了,到此,Sping 和 SpringBoot 两种项目下的 Redis 高可用集群的哨兵模式(Redis-Sentinel) 已经验证完成,且结果完美。
Redis 进击物语:
Redis进击(一)从0到1,Redis的安装与使用
Redis进击(二)搭建Redis主从复制服务集群(一主两从、反客为主)【Windows环境】
Redis进击(三)搭建Redis高可用集群的哨兵模式(Redis-Sentinel)【Windows环境】
Redis进击(四)Java中配置和使用Redis高可用集群的哨兵模式(Redis-Sentinel)【Spring&SpringBoot环境】
Redis进击(五)redis.conf配置文件说明备注手册
Redis异常:Creating Server TCP listening socket *:26379: bind : No such file or directory
Redis异常:JedisException: Can connect to sentinel, but 127.0.0.1:6379 seems to be not monitored...
Redis异常:JedisConnectionException: All sentinels down, cannot determine where is mymaster master is