分布式锁(Redis实现)
1.分布式锁解决方案
1.采用数据库 不建议 性能不好 jdbc
2.基于Redis实现分布式锁(setnx)setnx也可以存入key,如果存入key成功返回1,如果存入的key已经存在了,返回0.
3.基于Zookeeper实现分布式锁 Zookeeper是一个分布式协调工具,在分布式解决方案中。
多个客户端(jvm),同时在zk上创建相同的一个临时节点,因为临时节点路径是保证唯一,只要谁能够创建节点成功,谁就能够获取到锁,没有创建成功节点,就会进行等待,当释放锁的时候,采用事件通知给客户端重新获取锁的资源。
Redis实现分布式锁与Zookeeper实现分布式锁区别
实现分布式锁最终是通过什么方式?(相同点)
在集群环境下,保证只允许有一个jvm进行执行。
从技术上分析(区别)
Redis 是nosql数据,主要特点缓存
Zookeeper是分布式协调工具,主要用于分布式解决方案
实现思路( 区别)
核心通过获取锁、释放锁、死锁问题
获取锁
Zookeeper
多个客户端(jvm),会在Zookeeper上创建同一个临时节点,因为Zookeeper节点命名路径保证唯一,不允许出现重复,只要谁能够先创建成功,谁能够获取到锁。
Redis
多个客户端(jvm),会在Redis使用setnx命令创建相同的一个key,因为Redis的key保证唯一,不允许出现重复,只要谁能够先创建成功,谁能够获取到锁。
释放锁
Zookeeper使用直接关闭临时节点session会话连接,因为临时节点生命周期与session会话绑定在一块,如果session会话连接关闭的话,该临时节点也会被删除。
这时候客户端使用事件监听,如果该临时节点被删除的话,重新进入盗获取锁的步骤。
Redis在释放锁的时候,为了确保是锁的一致性问题,在删除的redis 的key时候,需要判断同一个锁的id,才可以删除。
共同特征:如何解决死锁现象问题
Zookeeper使用会话有效期方式解决死锁现象。
Redis 是对key设置有效期解决死锁现象
性能角度考虑
因为Redis是NoSQL数据库,相对比来说Redis比Zookeeper性能要好。
可靠性
从可靠性角度分析,Zookeeper可靠性比Redis更好。
因为Redis有效期不是很好控制,可能会产生有效期延迟,Zookeeper就不一样,因为Zookeeper临时节点先天性可控的有效期,所以相对来说Zookeeper比Redis更好
Redis实现的分布式锁,setnx可以存入key,如果存入key成功返回1,如果存入的key已经存在了返回0
使用setnx命令方式,同时在redis上创建相同的一个key。redis不允许重复key。创建成功的jvm获取锁,创建失败的jvm就等待。
写入时候 有key 返回1 没有key返回0
在redis中,key 是唯一的!
set持续的输入 key 的value会一直被覆盖哦 返回值是 ok
setnx key不存在1 存在0
删除之后,又可以了哈!
如何释放锁?
执行完操作时候,删除key。 但是 如果此时 redis挂了,或者删除失败,咋办? 解决办法是 给key设置有效期!!!防止死锁问题 (如果代码一直执行不完了,也可以搞定他!)
综述:执行完,删除key,每个key,都有期
多台服务器集群中,只能保证一个jvm进行操作!
identifierValue的作用! 防止a线程删除了 b线程刚刚获取到的锁!!自己删除自己的
赶紧上代码,注解已经很详细了~
pom.xml
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.toov5.redisLock</groupId>
<artifactId>redisLock</artifactId>
<version>0.0.1-SNAPSHOT</version>
<dependencies>
<dependency>
<groupId>redis.clients</groupId>
<artifactId>jedis</artifactId>
<version>2.9.0</version>
</dependency>
</dependencies> </project>
redis锁
package com.toov5; import java.util.UUID; import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool; //基于redis实现分布式锁,核心方法 获取锁、释放锁
public class LockRedis {
//redis线程池
private JedisPool jedisPool;
//需要创建的那个key
private String redisLockKey = "redis_lock";
//value是个不能重复的数字 锁的id public LockRedis(JedisPool jedisPool) {
this.jedisPool=jedisPool;
} /*
* @param acquiretimeOut 在获取锁之前超时
* @param timeOut 在获取锁之后超时
*/ public String getRedisLock(Long acquiretimeOut, Long timeOut){
Jedis conn = null;
try {
//1建立连接
conn=jedisPool.getResource();
//2、定义redis对应key的value (uuid生成) 释放锁时候会用到
String identifierValue = UUID.randomUUID().toString();
//3、reids实现分布式锁 有两个超时问题
//3.1获取锁之前,如果规定时间内 没有获取到锁 放弃
//3.2获取锁之后,可以对应有效期。规定时间内 失效
//4使用循环机制 保证重复进行尝试获取锁(乐观锁)获取不到 则放弃
//5 使用setnx命令插入对应的redislockkey,判断返回值进行业务
int expireLock =(int)(timeOut/1000); //以秒为单位 转换
Long endTime = System.currentTimeMillis()+acquiretimeOut;
while (System.currentTimeMillis()<endTime) { //小于结束时间
//则 获取锁
//6、使用setnx命令插入redislockkey,判断返回值
if (conn.setnx(redisLockKey, identifierValue)==1) {
//设置对应key的有效期
conn.expire(redisLockKey, expireLock); //时间为int类型的
return identifierValue; //插入成功返回 对应的值 规定时间内去循环获取
}
} } catch (Exception e) {
e.printStackTrace();
}finally{
if (conn != null) {
conn.close();
}
}
return null; //如果大于了 时间 就放弃了 返回 null 结束掉 } //释放redis锁
public void unRedisLock(String identifierValue){
//两种 1、执行完毕删除
//如果直接删除,有可能a刚刚获取,却被b删除了 所以保证是自己创建的redislockkey,自己的
Jedis conn = null;
conn=jedisPool.getResource();
try {
//如果该锁的id等于identifierValue
if (conn.get(redisLockKey).equals(identifierValue)) {
System.out.println("释放锁"+Thread.currentThread().getName()+",identifierValue");
conn.del(redisLockKey);
}
} catch (Exception e) { }finally{
if (conn !=null) {
conn.close();
}
} } }
业务逻辑:
package com.toov5; import redis.clients.jedis.JedisPool;
import redis.clients.jedis.JedisPoolConfig; public class LockService {
private static JedisPool pool = null;
//redis 连接代码
static {
JedisPoolConfig config = new JedisPoolConfig();
// 设置最大连接数
config.setMaxTotal(200);
// 设置最大空闲数
config.setMaxIdle(8);
// 设置最大等待时间
config.setMaxWaitMillis(1000 * 100);
// 在borrow一个jedis实例时,是否需要验证,若为true,则所有jedis实例均是可用的
config.setTestOnBorrow(true);
pool = new JedisPool(config, "192.168.91.5", 9001, 3000);
}
//创建一个redis锁
private LockRedis lockRedis = new LockRedis(pool); //定义一个方法 演示rdis实现分布式锁
public void seckill(){
String identifierValue = lockRedis.getRedisLock(5000L, 5000L);//获取到一个随机的返回结果
if (identifierValue == null) {
System.out.println(Thread.currentThread().getName()+"获取时间超时,锁获取失败");
return;
}
System.out.println(Thread.currentThread().getName()+",获取锁成功,锁的id"+identifierValue); //释放锁
lockRedis.unRedisLock(identifierValue); //表示锁的id } }
线程:
package com.toov5; public class ThreadRedis extends Thread {
private LockService lockService; public ThreadRedis(LockService lockService) {
this.lockService=lockService;
} @Override
public void run() { lockService.seckill();
} }
测试:
package com.toov5; public class RedisLockTest { public static void main(String[] args) {
LockService lockService = new LockService();
for (int i = 0; i < 50; i++) {
new ThreadRedis(lockService).start();
} } }
运行结果:
zk也可以通过设置连接超时时间来防止死锁!
三种分布式对比
上面几种方式,哪种方式都无法做到完美。就像CAP一样,在复杂性、可靠性、性能等方面无法同时满足,所以,根据不同的应用场景选择最适合自己的才是王道。
从理解的难易程度角度(从低到高)
数据库 > 缓存 > Zookeeper
从实现的复杂性角度(从低到高)
Zookeeper >= 缓存 > 数据库
从性能角度(从高到低)
缓存 > Zookeeper >= 数据库
从可靠性角度(从高到低)
Zookeeper > 缓存 > 数据库
Redis实现分布式锁与Zookeeper实现分布式锁区别
使用redis实现分布式锁
redis中的set nx 命令,当key不存在时,才能在redis中将key添加成功,利用该属性可以实现分布式锁,并且redis对于key有失效时间,可以控制当某个客户端加锁成功之后挂掉,导致阻塞的问题。
使用Zookeeper实现分布式锁
多个客户端在Zookeeper上创建一个相同的临时节点,因为临时节点只能允许一个客户端创建成功,那么只要任意一个客户端创建节点成功,谁就成功的获取到锁,当释放锁后,其他客户端同样道理在Zookeeper节点。
总结:
相同: 都是保证集群环境下,只有几个jvm进行执行
不同: redis 是NoSQL数据库,主要特点是做缓存
Zookeeper 是分布式协调工具,主要用于分布式解决方案的
主要考虑到三个方面 获取锁 释放锁 死锁
1、获取锁 多个jvm,会在Zookeeper上面创建一个临时节点,谁先创建成功。 节点唯一性
在redis上使用setnx命令创建一个key,谁先创建成功。可以的唯一性
2、释放锁 使用直接关闭临时节点session会话连接。临时节点就会删除。其他客户端事件监听,进入获取锁的环节步骤。
为了保证一致性,一定要判断锁的id才可以删除key
3、 防止死锁: Zookeeper会话有效期设置
Redis的key过期时间
性能角度上: redis比zk要好一些。 毕竟redis是NoSQL数据库,内存中哦。
可靠性来说 zk好 以为redis 的有效期不好控制 可能延迟 看代码: 拿到锁了设置有效期 过程有延迟 zk的节点 先天性可控~
分布式锁(Redis实现)的更多相关文章
- 分布式锁--Redis小试牛刀
参考文章: Redis分布式锁的正确实现方式 分布式锁看这篇就够了 在这两篇文章的指引下亲测 Redis分布式锁 引言 分布式系统一定会存在CAP权衡问题,所以才会出现分布式锁 什么是CAP理论? 为 ...
- 分布式锁----Redis实现
分布式锁 为什么需要有分布式锁呢,在单点的时候synchronized 就能解决,但是服务拆分之后,每个服务都是单独的机器,无法解决,所以出现了分布式锁,其实也就是用各种手段,实现获取唯一锁,别人无法 ...
- Redis除了做缓存--Redis做消息队列/Redis做分布式锁/Redis做接口限流
1.用Redis实现消息队列 用命令lpush入队,rpop出队 Long size = jedis.lpush("QueueName", message);//返回存放的数据条数 ...
- java-spring基于redis单机版(redisTemplate)实现的分布式锁+redis消息队列,可用于秒杀,定时器,高并发,抢购
此教程不涉及整合spring整合redis,可另行查阅资料教程. 代码: RedisLock package com.cashloan.analytics.utils; import org.slf4 ...
- 分布式锁redis
1. 首先看这篇文章中 https://mp.weixin.qq.com/s/s-ozSjM5WmSUopxttSWYeQ 为什么redis能实现锁功能呢,看下图,redis命令窗口中,setnx ...
- 分布式锁-Redis方案
#!/usr/bin/env python # coding=utf-8 import time import redis class RedisLock(object): def __init__( ...
- Lua脚本在redis分布式锁场景的运用
目录 锁和分布式锁 锁是什么? 为什么需要锁? Java中的锁 分布式锁 redis 如何实现加锁 锁超时 retry redis 如何释放锁 不该释放的锁 通过Lua脚本实现锁释放 用redis做分 ...
- Redis 分布式锁的实现
0X00 测试环境 CentOS 6.6 + Redis 3.2.10 + PHP 7.0.7(+ phpredis 4.1.0) [root@localhost ~]# cat /etc/issue ...
- 分布式交易系统的并发处理, 以及用Redis和Zookeeper实现分布式锁
交易系统 交易系统的数据结构 支付系统API通常需要一个“订单号”作为入参, 而实际调用API接口时使用到的往往不是真正意义的业务订单号, 而是交易订单号. 支付系统的API会使用“商户号+订单号” ...
- 基于zookeeper或redis实现分布式锁
前言 在分布式系统中,分布式锁是为了解决多实例之间的同步问题.例如master选举,能够获取分布式锁的就是master,获取失败的就是slave.又或者能够获取锁的实例能够完成特定的操作. 目前比较常 ...
随机推荐
- Codeforces A. Bear and Big Brother
...不行.这题之后.不做1000分以下的了.很耻辱 A. Bear and Big Brother time limit per test 1 second memory limit per t ...
- Extjs grid 单元格编辑
实现grid勾选后出现编辑按钮,通过增加一个字段checked来控制 事件如下: selectionchange: function (thi, selected, eOpts) { for (var ...
- MySQL中limit使用动态参数的解决方法(拼接SQL字符串语句来执行SQL)
官方好像说过limit已经在5.6版本上支持了动态参数,但是测试时依然还是不行. 那么要解决limit动态参数唯一能做的就是使用字符串SQL拼接的形式,然后再进行执行. 一般有以下方式解决: 1.存储 ...
- NSThread学习
使用多线程可以防止主线程阻塞.同时也可以将一个大的任务分成若干个小的任务去做. 常用方法一: 1, 首先使用 detachNewThreadSelector:toTarget:withObject: ...
- [IOS笔记] - 动画animation
//移动 - (IBAction)translation:(id)sender { CABasicAnimation *traslation = [CABasicAnimation animation ...
- 邁向IT專家成功之路的三十則鐵律 鐵律九:IT人社群互動之道-縮小自己
身為一位專業的IT人士所要學習的東西實在非常的多,然而對於時間相當有限的我們,最快速的學習方法就是向他人學習,而向他人學習的首要態度就是「縮小自己」.唯有將自己縮小到別人的眼睛裡,才能夠讓他們真心誠意 ...
- 转:Android IOS WebRTC 音视频开发总结 (系列文章集合)
随笔分类 - webrtc Android IOS WebRTC 音视频开发总结(七八)-- 为什么WebRTC端到端监控很关键? 摘要: 本文主要介绍WebRTC端到端监控(我们翻译和整理的,译 ...
- Leet Code OJ 338. Counting Bits [Difficulty: Medium]
题目: Given a non negative integer number num. For every numbers i in the range 0 ≤ i ≤ num calculate ...
- table 设置边框
本文引自:https://www.cnblogs.com/leona-d/p/6125896.html 示例代码: <!DOCTYPE html> <html lang=" ...
- GOOGLE VR SDK开发VR游戏,VR播放器之中的一个
近期一年来,VR虚拟现实和AR增强现实技术的宣传甚嚣尘上.事实上VR,AR技术非常早就有了,一直没有流行开来.不可否认价格是影响技术推广的最大壁垒. 谷歌对VR最大的贡献是提供了便宜的谷歌眼镜,依照G ...