redis分片:
redis如果采用分片机制,将来如果有一台宕机,则直接影响程序的正常运行

如何解决分片缺点:
1.实现高可用,当redis服务器宕机时可以自动的实现主从切换
2.缓存数据同步是实现高可用的前提
3. 如果没有数据同步,即使实现了高可用,则也可能由于从服务器中没有数据而导致缓存雪崩效应

redis分片特点:
1.redis分片机制可以实现redis内存数据的扩容
2.redis分片机制中,是业务服务器进行一致性hash的`计算,而reids服务器只需要负责数据的存储即可,所以redis分片机制性能更高
3.redis分片机制本身没有实现高可用的效果,如果redis节点缺失,则直接影响用户的使用

redis主从搭建
要求:实现主从同步

主从测试
1.当搭建主从结构之后,用户操作主机,从机可以自动实现数据同步
2.搭建主从结构后,从机只读操作,不能写入
3.不要使用主从结构,测试分片API

主从配置有效期
1.默认条件下通过slaveof指令,制定主从结构只能在内存中有效,如果服务器关机重启,则该结构失效,需要手动修改
2.如需要指定永久配置,则需要修改配置文件redis.conf,但是该文件不能由用户自己操作,最好由哨兵/集群的策略完成

redis哨兵工作原理
1.哨兵启动启动后,首先监控当前redis的主机,并且从主机中其获得从机的信息
2.当redis主机宕机之后,哨兵通过心跳检测机制检测主机是否宕机,如果连续3次都没有获取主机的反馈,则断定主机宕机,之前根据算法筛选出 新的主机
3.当哨兵选举出新的主机后,为了保证主从的关系,则会动态的修改各自的redis.conf文件,并且将其他节点标识为新主机的从机

redis哨兵特点:
1.实现redis节点的高可用
2.redis哨兵机制不能实现内存数据的扩容
3.哨兵本身没有实现高可用,哨兵如果宕机,则直接影响用户使用

为什么要使用redis集群??
1.为了提高网站响应速度,总是把热点数据保存在内存中而不是直接从后端数据库中读取
2.由于内存大小的限制,使用一台 Redis 实例显然无法满足需求,这时就需要使用多台 Redis作为缓存数据库。但是如何保证数据存储的一致性呢,这时就需要搭建redis集群.采用合理的机制,保证用户的正常的访问需求.
3.采用redis集群,可以保证数据分散存储,同时保证数据存储的一致性.并且在内部实现高可用的机制.实现了服务故障的自动迁移.

如何进行redis集群高可用测试?
将主机宕机,等待15秒后检查是否有从机当选主机,然后重启主机,检查是否成为新主机的从机

如果redis集群搭建失败该怎么操作?
1.关闭redis的所有服务器
2.删除所有的文件,只保留redis.conf文件

[root@localhost cluster]# rm -rf 700*/dump.rdb
[root@localhost cluster]# rm -rf 700*/nodes.conf

3.重启redis
sh start.sh

4.重新搭建集群的指令

redis-cli --cluster create --cluster-replicas 1 
192.168.126.129:7000 192.168.126.129:7001 
192.168.126.129:7002 192.168.126.129:7003 
192.168.126.129:7004 192.168.126.129:7005

面试题
1.redis集群中最多存储16384个 错误
解释:redis的hash槽算法,只能判断某个数据属于哪台管理,至于数据存储多少个,取决于redis的内存决定
crc16(“key1”)%16384=1000
crc16(“key2”)%16384=1000
计算只能哪块存储,

2.redis集群中最多有多少台主机??? 16384台主机
一块区域由一个redis管理. 一般工作中使用30-80 够用即可.

编辑redis的properties文件

#配置redis单台
#redis.host=192.168.126.129
#redis.port=6379

#配置redis分片机制
#redis.shards=192.168.126.129:6379,192.168.126.129:6380,192.168.126.129:6381

#配置redis集群
redis.cluster=192.168.126.129:7000,192.168.126.129:7001,192.168.126.129:7002,192.168.126.129:7003,192.168.126.129:7004,192.168.126.129:7005

编辑配置类(redis集群)

@Configuration  //标识配置类  一般和@Bean注解联用
@PropertySource("classpath:/properties/redis.properties")
public class RedisConfig {
    
    @Value("${redis.cluster}")
    private String nodes;   //node,node,node
    
    //配置redis集群
    @Bean
    public JedisCluster   jedisCluster() {
        //1.定义redis链接池
        JedisPoolConfig poolConfig = new JedisPoolConfig();
        poolConfig.setMaxTotal(1000);  //线程存活时间
        poolConfig.setMaxIdle(20);  //最大空闲数量
        
        //2.准备redis节点信息
        Set<HostAndPort> set = new HashSet<>();
        String[] nodeArray = nodes.split(","); //分割节点信息
        for (String node : nodeArray) {   //host:port
            String host = node.split(":")[0];  //[0] 指的是分隔符之前
            int port = Integer.parseInt(node.split(":")[1]); //[1]指的是分隔符之后
            set.add(new HostAndPort(host, port));
        }
        return new JedisCluster(set, poolConfig);
    }

编辑CacheAOP(redis集群)

@Aspect   //使该类成为切面类
@Component  //把切面类加入到IOC容器中 
public class CacheAOP {

	@Autowired(required = false)  //  使用到该依赖注入才起效
	private JedisCluster jedis;  //集群
	//private ShardedJedis jedis;  //引入分片机制
	//private Jedis jedis;  //单台

	/**
	 * 规定:ProceedingJoinPoint  参数必须位于第一位
	 * 1,利用切入点表达式拦截@CacheFind注解,同时获取CacheFind注解对象
	 * 2.自定义key
	 * 3.根据key查询缓存
	 *   有数据,则返回对象
	 *   没有数据,则查询数据库,之后保存到缓存
	 *   
	 *   反射相关:
	 * 
	 *  Method method = 
                joinPoint.getTarget()   //获取目标对象
                         .getClass()    //目标对象的class
                         .getMethod("findItemCatList", Long.class); //方法
                //从方法对象中获取返回值类型
                Class returnClass = method.getReturnType();
	 * @throws Throwable 
	 */
	@SuppressAjWarnings("unchecked")  //压制警告注解  unchecked行了未检查的转换时的警告
	//@Around("@annoation(com.jt.anno.cacheFind)")
	@Around("@annotation(cacheFind)")
	public Object around(ProceedingJoinPoint joinPoint,CacheFind cacheFind){
		Object result = null;
		System.out.println("环绕通知开始");
		try {
			Object[] ages = joinPoint.getArgs(); //目标方法参数
			String key = cacheFind.prekey()+"::"+Arrays.toString(ages);
			//根据key查询redis
			if(!jedis.exists(key)) {
				//如果key不存在,则执行目标方法

				result = joinPoint.proceed();
				System.out.println("查数据库");

				//调用业务方法
				//将数据存入缓存中redis
				String json = ObjectMapperUtil.toJSON(result);

				int seconds = cacheFind.seconds();
				if (seconds>0)
					//表示用户需要设置超时时间
					jedis.setex(key, seconds, json);
				else
					//表示用户不需要设置超时时间
					jedis.setnx(key, json);
			}else {
				//如果redis中有缓存,则将json串转化为对象之后返回
				String json = jedis.get(key);//获取json
				//如何获取方法的返回值类型?先获取方法对象
				MethodSignature methodSignature = (MethodSignature)joinPoint.getSignature();//获取对象的方法
				result = ObjectMapperUtil.toObj(json, methodSignature.getReturnType());		
				System.out.println("AOP缓存");
			}
		}catch(Throwable e) {
			e.printStackTrace();
		}
		System.out.println("环绕通知结束");
		return result;
	}
	@Around("@annotation(cacheUpdate)")
	//@Around("@annoation(com.jt.anno.cacheUpdate)")
	public Object updateAround(ProceedingJoinPoint joinPoint,CacheUpdate cacheUpdate) {
		/**
		 * 缓存更新AOP实现 首选环绕通知
		 * 要求:
		 * 执行目标方法
		 * 将返回值更新redis缓存
		 */
		try {
			//暂时写死
			ItemCat itemCat = (ItemCat)joinPoint.proceed();
			String key = cacheUpdate.prekey()+"::+["+itemCat.getId()+"]";
			String json = ObjectMapperUtil.toJSON(itemCat);
			if (cacheUpdate.seconds()>0) {
				//表示用户需要设置超时时间
				jedis.setex(key, cacheUpdate.seconds(), json);
			}else {
				jedis.set(key, json);
			}
			return itemCat;
		} catch (Throwable e) {
			e.printStackTrace();
			throw new RuntimeException(e);
		}
	}
}