Redis实现分布式锁分三个步骤:1、获取锁 2、操作资源 3、释放锁 。看似简单,其中有很多细节需处理
1、在释放锁之前,抛异常了,怎么办:锁是一定要释放的。不释放,其它线程就不可能拿到这个锁,执行相关操作。使用try{}finally{},在finally中释放锁。
2、在释放锁之前,程序意外退出 了或者宕机了,怎么办:配合Expire命令,增加超时时间, 这样就能在锁超时的时候释放掉锁。
3、setNX与设置超时时间不是原子性的,如果宕机发生在获取锁与设置超时时间之间,怎么办:可以使用 Set Nx Ex命令(代码中用setIfAbsent)
4、锁失效【线程B加的锁,被线程A释放掉了】的问题怎么解决(线程A任务还没执行完,锁设置的过期时间已经超时,那么其它线程就能获取到锁进行相关操作,那么此时线程A执行完释放的锁就是其它线程的锁了):给每个锁设置唯一的Id,在删除之前先判断是否属于自己那把锁。
5、锁超时了,任务还没执行完怎么办:给锁续命。开启一个子线程,每隔超时时间的三分之一,就去检查当前锁是否还存在,如果存在说明还没释放也就是任务还没执行完,就进行续约。如果当前锁不存在了,说明锁已经释放了,该子线程也可以销毁了。
写在最后:Java开发菜鸟一枚,面试中Redis分布式锁被问到的机率是很大的,在此做下总结。如果错误不足之处,还望指正。