Redis实现写入时切换库的功能

这里的实现,是通过调用Redis的工具类,通过Spring的AOP来实现切换库。

首先,我们先定义一个注解,用于填加在想要切换库的方法上。

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface RedisSelect {
    /**
     * redis库   0 - 15  库
     * @return
     */
    int value() default 0;
}

创建对应的切面,来对标有注解的方法拦截

    /**
     * 环绕增强标注 RedisSelect 注解的方法
     * @author Mrlv
     * @date 2021/12/19 16:15
     * @param point
     * @return java.lang.Object
     */
    @Around("@annotation(com.mrlv.redis.annotation.RedisSelect)")
    @ConditionalOnBean(SelectableRedisTemplate.class)
    public Object configRedis(ProceedingJoinPoint point) throws Throwable{
        int db = defaultDataBase;
        try {
            MethodSignature signature = (MethodSignature) point.getSignature();
            Method method = signature.getMethod();
            RedisSelect config = method.getAnnotation(RedisSelect.class);
            if(config != null){
                db = config.value();
            }
            RedisSelectSupport.select(db);
            return point.proceed();
        } finally {
            //在环绕通知执行回调完成后 重置 RedisSelectSupport 中的静态 db
            RedisSelectSupport.select(defaultDataBase);
            //logger.debug("redis 重置db {} 到 {}", db, defaultDataBase);
        }
    }

然后我们创建一个类,定义一个静态变量 (TransmittableThreadLocal可以自己去了解,简单来说解决线程池传递值的问题)。

/**
 * Redis 切换DB配置
 */
public class RedisSelectSupport {

    private static final TransmittableThreadLocal<Integer> SELECT_CONTEXT = new TransmittableThreadLocal<>();

    public static void select(int db) {
        SELECT_CONTEXT.set(db);
    }

    public static Integer getSelect() {
        return SELECT_CONTEXT.get();
    }
}

接下来自定义SelectableRedisTemplate类继承并重写RedisTemplate的preProcessConnection方法。

public class SelectableRedisTemplate<K, V> extends RedisTemplate<K, V> {

    @Override
    protected RedisConnection createRedisConnectionProxy(RedisConnection pm) {
        return super.createRedisConnectionProxy(pm);
    }

    /**
     * 在连接Redis之前做一些配置
     * @param connection
     * @param existingConnection
     * @return
     */
    @Override
    protected RedisConnection preProcessConnection(RedisConnection connection, boolean existingConnection) {
        Integer db = RedisSelectSupport.getSelect();
        if(db != null){
            //切换 redis db 到 其他的库
            connection.select(db);
        }
        return super.preProcessConnection(connection, existingConnection);
    }
}

重写注入RedisTemplate方法

    /**
     * 实例化RedisTemplate对象
     * @return
     */
    @Bean
    @ConditionalOnMissingBean
    public SelectableRedisTemplate<String, Object> functionDomainRedisTemplate(RedisConnectionFactory redisConnectionFactory, RedisSerializer redisMyStringSerializer) {
        SelectableRedisTemplate<String, Object> redisTemplate = new SelectableRedisTemplate<String, Object>();
        redisTemplate.setConnectionFactory(redisConnectionFactory);
        redisTemplate.afterPropertiesSet();
        log.info("实例化 RedisTemplate 对象完成");
        return redisTemplate;
    }

即可。

使用方法

@RedisSelect(4)添加在对应的方法体上即可。
  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Redis Cluster中,主从切换可能会导致数据丢失。当集群发生故障或分区,Sentinel集群会重新选举一个新的Master节点来替代故障的Master节点。然而,在切换过程中可能会发生数据丢失的情况。 首先,在某种情况下,比如网络原因导致Master与Slave节点之间失去联系,Sentinel集群可能会错误地认为Master节点故障并进行切换。这,旧的Master节点可能并没有真正发生故障,而是由于网络分区导致无法与其他节点通信。如果不及发现问题并进行处理,旧的Master节点可能会继续接收客户端的写入请求,而新的Master节点并不包含这些数据。这样就会导致旧的Master节点中堆积大量数据,当问题被发现后,旧的Master节点需要降级为Slave来同步新的Master节点的数据,这样之前堆积的数据就会被刷新掉,造成数据丢失。 其次,集群产生脑裂也可能导致数据丢失。脑裂是指集群中的节点无法达成一致,导致部分节点认为Master节点故障,而另一部分节点认为原Master节点仍然存活。在这种情况下,两个Master节点可能同接收写入请求,导致数据不一致。 为了尽量减少数据丢失,可以采取一些措施。首先,可以使用持久化机制来定期将数据写入磁盘,以防止节点宕机后数据丢失。然而,即使开启持久化设置,当Master节点发生故障并被切换后,旧的Master节点在故障恢复后重启,仍然需要同步新的Master节点的数据,此旧的Master节点中的数据会被刷新掉,仍然会造成数据丢失。 因此,在Redis Cluster中,无法完全保证数据不丢失,只能尽量减少数据丢失的风险。对于高可用性的要求,可以采用更可靠的数据备份和灾难恢复措施,以确保数据的安全性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值