赞
踩
一定要结合业务场景来回答问题!要是没有不要硬讲,除非面试官问;
接下来面试官将深入发问。
Redis 分布式锁使用的场景?
分布式情况下的,或者集群情况下的,多个微服务操作同一对象,可能是相同操作(同时改),也可能是不同操作(一个删,一个改…)
执行流程:
而如果是分布式的项目,就可能对服务进行集群部署:
显然这解决不了分布式问题;
只能解决同个 Java 进程的多线程的问题,也就是处理同一个 Java 服务的多个请求的问题;
Redis 分布式锁其实就是“争夺一个 key 的定义”,主要利用 Redis 的 setnx 命令(SET if **not ex**ists 如果不存在,则设置)
小插曲:锁提前过期,但是业务未结束,锁岂不是可以被其他服务获取了?
其实这个问题可以通过给锁续期来解决,举个例子:
一个分布式锁有效时长是 5 秒,但是业务时长 7 秒,我们可以每隔一段时间,如每个 3 秒 就判断业务是否结束,如果结束了那就释放锁,如果没结束,就重置锁的有效时长,如重置为 5 秒;
分布式锁真的很灵活和精准!
可以使用 redission 框架(加锁、设置过期时间等操作都是基于 lua 脚本,Redis可以识别的可保证原子性的脚本完成),执行流程如下:
示例代码:
RLock 锁是可以重入的锁
正常情况下,Redis 集群的主从会进行同步,所以不同的 Java 应用访问 Redis 集群的时候,是只有一个应用能获取一把锁;
但是,如果主节点在从节点同步之前挂了,从节点就变成了新的主节点,另一个 Java 应用尝试获取这把锁,发现可以获取成功,就出现了两个线程同时持有一把锁的情况了;
之后,两个线程就没有阻塞的同时进行了。
主从就不一致了,因为从节点只显示正常一个线程获得锁,但实际上是两个线程用了一把锁;
可以使用分布式锁的另一种实现:RedLock(**Red**is)
RedLock(红锁),在多个 Redis 实例上创建锁 (n / 2 + 1),避免在一个 Redis 实例上加锁:
缺点:
如果非要解决这个问题,可以用 zookeeper(CP 思想)去解决,而 Redis (AP 思想)更适合根据具体业务实现最终一致性;
Redis 分布式锁,是如何实现的?
- 根据自己简历的项目进行描述分布式锁使用后的场景,如抢票、维持幂等性…
- 我们当时使用的是 redisson 实现的分布式锁,底层是 setnx 和 lua 脚本(保证原子性)
Redisson 实现的分布式锁是如何合理的控制锁的有效时长的?
- 在 redisson 的分布式锁中,提供了一个 ==WatchDog(看门狗)==一个线程获得锁成功以后 WatchDog 会给持有锁的线程**续期(默认是每个 10 秒续期一次)**
Redisson 的这个锁,是可重入锁么?
- 可以重入,多个锁重入需要判断是否是当前线程,在 Redis 中进行存储的时候使用 hash 结构,来存储==线程信息和重入的次数==;
Redisson 的这个锁,能处理主从数据一致的问题吗?
主从数据一致的问题:主节点在从节点同步之前宕机,从节点就被升级为主节点的数据不一致的问题;
处理不了,但是可以用 Redisson 提供的的==红锁来解决,但是这样的话,性能太低了==,如果是 AP,则为我们保证数据的最终一致性即可,如果业务要求 CP,建议采用 zookeeper 实现的分布式锁;
在 Redis 中提供了三种集群方案:
有一些常见的问题:
主从同步原理:
由于单点 Redis 的并发能力是有上限的,要进一步提高 Redis 的并发能力,就需要搭建主从集群,实现读写分离;
主从全量同步:
主从增量同步(slave 重启或后期数据变化)
主从同步数据的流程:
Redis 提供了哨兵(Sentinel)机制来实现主从集群的自动故障恢复。哨兵的结构和作用如下:
状态监控的相关细节:
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。