Redis的相关知识在分布式架构中的应用

本篇文章给大家带来了关于Redis的相关知识,在分布式架构中,我们同样会遇到数据共享操作问题,使用Redis来解决分布式架构中的数据一致性问题,下面一起来看一下,希望对大家有帮助。

1. 单机数据一致性

单机数据一致性架构如下图所示:多个可客户访问同一个服务器,连接同一个数据库。

场景描述:客户端模拟购买商品过程,在Redis中设定库存总数剩100,多个客户端同时并发购买。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

@RestController

publicclassIndexController1 {

@Autowired

StringRedisTemplate template;

@RequestMapping("/buy1")

publicString index(){

// Redis中存有goods:001号商品,数量为100

String result = template.opsForValue().get("goods:001");

// 获取到剩余商品数

inttotal = result == null0: Integer.parseInt(result);

if( total > 0){

// 剩余商品数大于0 ,则进行扣减

intrealTotal = total -1;

// 将商品数回写数据库

template.opsForValue().set("goods:001",String.valueOf(realTotal));

System.out.println("购买商品成功,库存还剩:"+realTotal +"件, 服务端口为8001");

return"购买商品成功,库存还剩:"+realTotal +"件, 服务端口为8001";

}else{

System.out.println("购买商品失败,服务端口为8001");

}

return"购买商品失败,服务端口为8001";

}

}

使用Jmeter模拟高并发场景,测试结果如下:

测试结果出现多个用户购买同一商品,发生了数据不一致问题!

解决办法:单体应用的情况下,对并发的操作进行加锁操作,保证对数据的操作具有原子性

  • synchronized

  • ReentrantLock

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

@RestController

publicclassIndexController2 {

// 使用ReentrantLock锁解决单体应用的并发问题

Lock lock = newReentrantLock();

@Autowired

StringRedisTemplate template;

@RequestMapping("/buy2")

publicString index() {

lock.lock();

try{

String result = template.opsForValue().get("goods:001");

inttotal = result == null0: Integer.parseInt(result);

if(total > 0) {

intrealTotal = total - 1;

template.opsForValue().set("goods:001", String.valueOf(realTotal));

System.out.println("购买商品成功,库存还剩:"+ realTotal + "件, 服务端口为8001");

return"购买商品成功,库存还剩:"+ realTotal + "件, 服务端口为8001";

else{

System.out.println("购买商品失败,服务端口为8001");

}

catch(Exception e) {

lock.unlock();

finally{

lock.unlock();

}

return"购买商品失败,服务端口为8001";

}

}

2. 分布式数据一致性

上面解决了单体应用的数据一致性问题,但如果是分布式架构部署呢,架构如下:

提供两个服务,端口分别为80018002,连接同一个Redis服务,在服务前面有一台Nginx作为负载均衡

两台服务代码相同,只是端口不同

80018002两个服务启动,每个服务依然用ReentrantLock加锁,用Jmeter做并发测试,发现会出现数据一致性问题!

3. Redis实现分布式锁

3.1 方式一

取消单机锁,下面使用redisset命令来实现分布式加锁

SET KEY VALUE [EX seconds] [PX milliseconds] [NX|XX]

  • EX seconds 设置指定的到期时间(以秒为单位)

  • PX milliseconds 设置指定的到期时间(以毫秒为单位)

  • NX 仅在键不存在时设置键

  • XX 只有在键已存在时才设置

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

@RestController

publicclassIndexController4 {

// Redis分布式锁的key

publicstaticfinalString REDIS_LOCK = "good_lock";

@Autowired

StringRedisTemplate template;

@RequestMapping("/buy4")

publicString index(){

// 每个人进来先要进行加锁,key值为"good_lock",value随机生成

String value = UUID.randomUUID().toString().replace("-","");

try{

// 加锁

Boolean flag = template.opsForValue().setIfAbsent(REDIS_LOCK, value);

// 加锁失败

if(!flag){

return"抢锁失败!";

}

System.out.println( value+ " 抢锁成功");

String result = template.opsForValue().get("goods:001");

inttotal = result == null0: Integer.parseInt(result);

if(total > 0) {

intrealTotal = total - 1;

template.opsForValue().set("goods:001", String.valueOf(realTotal));

// 如果在抢到所之后,删除锁之前,发生了异常,锁就无法被释放,

// 释放锁操作不能在此操作,要在finally处理

// template.delete(REDIS_LOCK);

System.out.println("购买商品成功,库存还剩:"+ realTotal + "件, 服务端口为8001");

return"购买商品成功,库存还剩:"+ realTotal + "件, 服务端口为8001";

else{

System.out.println("购买商品失败,服务端口为8001");

}

return"购买商品失败,服务端口为8001";

}finally{

// 释放锁

template.delete(REDIS_LOCK);

}

}

}

上面的代码,可以解决分布式架构中数据一致性问题。但再仔细想想,还是会有问题,下面进行改进。

3.2 方式二(改进方式一)

在上面的代码中,如果程序在运行期间,部署了微服务jar包的机器突然挂了,代码层面根本就没有走到finally代码块,也就是说在宕机前,锁并没有被删除掉,这样的话,就没办法保证解锁

所以,这里需要对这个key加一个过期时间,Redis中设置过期时间有两种方法:

  • template.expire(REDIS_LOCK,10, TimeUnit.SECONDS)

  • template.opsForValue().setIfAbsent(REDIS_LOCK, value,10L,TimeUnit.SECONDS)

第一种方法需要单独的一行代码,且并没有与加锁放在同一步操作,所以不具备原子性,也会出问题

第二种方法在加锁的同时就进行了设置过期时间,所有没有问题,这里采用这种方式

调整下代码,在加锁的同时,设置过期时间:

1

2

// 为key加一个过期时间,其余代码不变

Boolean flag = template.opsForValue().setIfAbsent(REDIS_LOCK,value,10L,TimeUnit.SECONDS);

这种方式解决了因服务突然宕机而无法释放锁的问题。但再仔细想想,还是会有问题,下面进行改进。

3.3 方式三(改进方式二)

方式二设置了key的过期时间,解决了key无法删除的问题,但问题又来了

上面设置了key的过期时间为10秒,如果业务逻辑比较复杂,需要调用其他微服务,处理时间需要15秒(模拟场

景,别较真),而当10秒钟过去之后,这个key就过期了,其他请求就又可以设置这个key,此时如果耗时15

的请求处理完了,回来继续执行程序,就会把别人设置的key给删除了,这是个很严重的问题!

所以,谁上的锁,谁才能删除

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

@RestController

publicclassIndexController6 {

publicstaticfinalString REDIS_LOCK = "good_lock";

@Autowired

StringRedisTemplate template;

@RequestMapping("/buy6")

publicString index(){

// 每个人进来先要进行加锁,key值为"good_lock"

String value = UUID.randomUUID().toString().replace("-","");

try{

// 为key加一个过期时间

Boolean flag = template.opsForValue().setIfAbsent(REDIS_LOCK, value,10L,TimeUnit.SECONDS);

// 加锁失败

if(!flag){

return"抢锁失败!";

}

System.out.println( value+ " 抢锁成功");

String result = template.opsForValue().get("goods:001");

inttotal = result == null0: Integer.parseInt(result);

if(total > 0) {

// 如果在此处需要调用其他微服务,处理时间较长。。。

intrealTotal = total - 1;

template.opsForValue().set("goods:001", String.valueOf(realTotal));

System.out.println("购买商品成功,库存还剩:"+ realTotal + "件, 服务端口为8001");

return"购买商品成功,库存还剩:"+ realTotal + "件, 服务端口为8001";

else{

System.out.println("购买商品失败,服务端口为8001");

}

return"购买商品失败,服务端口为8001";

}finally{

// 谁加的锁,谁才能删除!!!!

if(template.opsForValue().get(REDIS_LOCK).equals(value)){

template.delete(REDIS_LOCK);

}

}

}

}

这种方式解决了因服务处理时间太长而释放了别人锁的问题。这样就没问题了吗?

3.4 方式四(改进方式三)

在上面方式三下,规定了谁上的锁,谁才能删除,但finally快的判断和del删除操作不是原子操作,并发的时候也会出问题,并发嘛,就是要保证数据的一致性,保证数据的一致性,最好要保证对数据的操作具有原子性。

Redisset命令介绍中,最后推荐Lua脚本进行锁的删除,地址

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

@RestController

publicclassIndexController7 {

publicstaticfinalString REDIS_LOCK = "good_lock";

@Autowired

StringRedisTemplate template;

@RequestMapping("/buy7")

publicString index(){

// 每个人进来先要进行加锁,key值为"good_lock"

String value = UUID.randomUUID().toString().replace("-","");

try{

// 为key加一个过期时间

Boolean flag = template.opsForValue().setIfAbsent(REDIS_LOCK, value,10L,TimeUnit.SECONDS);

// 加锁失败

if(!flag){

return"抢锁失败!";

}

System.out.println( value+ " 抢锁成功");

String result = template.opsForValue().get("goods:001");

inttotal = result == null0: Integer.parseInt(result);

if(total > 0) {

// 如果在此处需要调用其他微服务,处理时间较长。。。

intrealTotal = total - 1;

template.opsForValue().set("goods:001", String.valueOf(realTotal));

System.out.println("购买商品成功,库存还剩:"+ realTotal + "件, 服务端口为8001");

return"购买商品成功,库存还剩:"+ realTotal + "件, 服务端口为8001";

else{

System.out.println("购买商品失败,服务端口为8001");

}

return"购买商品失败,服务端口为8001";

}finally{

// 谁加的锁,谁才能删除,使用Lua脚本,进行锁的删除

Jedis jedis = null;

try{

jedis = RedisUtils.getJedis();

String script = "if redis.call('get',KEYS[1]) == ARGV[1] "+

"then "+

"return redis.call('del',KEYS[1]) "+

"else "+

"   return 0 "+

"end";

Object eval = jedis.eval(script, Collections.singletonList(REDIS_LOCK), Collections.singletonList(value));

if("1".equals(eval.toString())){

System.out.println("-----del redis lock ok....");

}else{

System.out.println("-----del redis lock error ....");

}

}catch(Exception e){

}finally{

if(null!= jedis){

jedis.close();

}

}

}

}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值