首頁 > 資料庫 > Redis > redis會發生死鎖問題嗎

redis會發生死鎖問題嗎

發布: 2019-07-08 09:14:05
原創
5707 人瀏覽過

redis會發生死鎖問題嗎

就分散式鎖定而言,一個常用的問題就是如果一個服務setnx成功了,但是在解鎖的時候如果發生了宕機或者一些特殊因素,導致無法解鎖,那麼其他服務將陷入死鎖的狀態。所以,我們在用setnx 的同時想著去用expire 指令對鎖進行一個過期操作, 從指令可以看出setnx 和expire指令是分開的,如果在這中間的空隙過程中如果有特殊因素導致指令無法繼續,也會導致死鎖的產生。

解決方法:

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

64

65

import org.slf4j.Logger;

import org.slf4j.LoggerFactory;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.data.redis.core.StringRedisTemplate;

import org.springframework.stereotype.Component;

import org.springframework.util.StringUtils;

  

@Component

public class RedisLock {

  

    Logger logger = LoggerFactory.getLogger(this.getClass());

  

    @Autowired

    private StringRedisTemplate redisTemplate;

  

    /**

     * 加锁

     * @param key  

     * @param value 当前时间 + 超时时间

     * @return

     */

    public boolean lock(String key, String value) {

     

        if (redisTemplate.opsForValue().setIfAbsent(key, value)) {    

            // 这个其实就是setnx命令,只不过在java这边稍有变化,返回的是boolean

            // 设置个过期时间,当然如果在这中间的空隙过程中如果有特殊因素导致指令无法继续,也会导致死锁的产生,如果死锁出现,则后续代码会处理

            redisTemplate.expire(key, lockTime, TimeUnit.SECONDS);

            return true;

        }

  

        // 避免死锁,且只让一个线程拿到锁

        String currentValue = redisTemplate.opsForValue().get(key);

        // 如果锁过期了

        if (!StringUtils.isEmpty(currentValue) && Long.parseLong(currentValue) < System.currentTimeMillis()) {

            //获取上一个锁的时间

            String oldValues = redisTemplate.opsForValue().getAndSet(key, value);

  

            /*

               只会让一个线程拿到锁

               如果旧的value和currentValue相等,只会有一个线程达成条件,因为第二个线程拿到的oldValue已经和currentValue不一样了

             */

            if (!StringUtils.isEmpty(oldValues) && oldValues.equals(currentValue)) {

                return true;

            }

        }

        return false;

    }

  

  

    /**

     * 解锁

     * @param key

     * @param value

     */

    public void unlock(String key, String value) {

        try {

            String currentValue = redisTemplate.opsForValue().get(key);

            if (!StringUtils.isEmpty(currentValue) && currentValue.equals(value)) {

                redisTemplate.opsForValue().getOperations().delete(key);

            }

        } catch (Exception e) {

            logger.error("redis分布式锁解锁异常,{}", e);

        }

    }

}

登入後複製

呼叫:

1

2

3

4

5

6

7

8

9

10

11

12

13

//加锁

   long time = System.currentTimeMillis() + 1000 * lockTime //超时时间:10秒,最好设为常量

 

   boolean isLock = redisLock.lock(...keyName, String.valueOf(time));

   if(!isLock){

       throw new RuntimeException("系统正忙");

   }

    

   // doSomething...

    

    

   //解锁

   redisLock.unlock(...keyName, String.valueOf(time));

登入後複製

更多Redis相關知識,請造訪Redis使用教學欄位!

以上是redis會發生死鎖問題嗎的詳細內容。更多資訊請關注PHP中文網其他相關文章!

相關標籤:
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板