-
Notifications
You must be signed in to change notification settings - Fork 23
/
Copy pathRedisLockTool.java
137 lines (122 loc) · 4.58 KB
/
RedisLockTool.java
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
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.script.DefaultRedisScript;
import org.springframework.stereotype.Component;
import java.util.Collections;
import java.util.concurrent.TimeUnit;
/**
* @author wenjie
* @date 2018/5/7 0007 16:44
*/
@Component
public final class RedisLockTool {
private static final Long SUCCESS = 1L;
public static final String LOCK_SCRIPT_STR = "if redis.call('set',KEYS[1],ARGV[1],'EX',ARGV[2],'NX') then return 1 else return 0 end";
public static final String UNLOCK_SCRIPT_STR = "if redis.call('get', KEYS[1]) == ARGV[1] then return redis.call('del', KEYS[1]) else return 0 end";
//default value
public static final Integer DEFAULT_EXPIRE_SECOND = 60;
public static final Long DEFAULT_LOOP_TIMES = 10L;
public static final Long DEFAULT_SLEEP_INTERVAL = 500L;
public static final String PACKAGE_NAME_SPLIT_STR = "\\.";
public static final String CLASS_AND_METHOD_CONCAT_STR = "->";
private static RedisTemplate redisTemplate;
@Autowired
public void init(RedisTemplate redisTemplate) {
this.redisTemplate = redisTemplate;
}
/**
* 得到分布式锁
* 默认key:调用者类名
*
* @return
* @throws InterruptedException
*/
public static boolean tryGetDistributedLock() {
String callerKey = getCurrentThreadCaller();
String requestId = String.valueOf(Thread.currentThread().getId());
return tryGetDistributedLock(callerKey, requestId);
}
/**
* @param lockKey 锁名称
* @param requestId 随机请求id
* @return
* @throws InterruptedException
*/
public static boolean tryGetDistributedLock(String lockKey, String requestId) {
return tryGetDistributedLock(lockKey, requestId, DEFAULT_EXPIRE_SECOND);
}
/**
* @param lockKey key
* @param requestId 随机请求id
* @param expireSecond 超时秒
* @return
* @throws InterruptedException
*/
public static boolean tryGetDistributedLock(String lockKey, String requestId, Integer expireSecond) {
return tryGetDistributedLock(lockKey, requestId, expireSecond, DEFAULT_LOOP_TIMES, DEFAULT_SLEEP_INTERVAL);
}
/**
* 加锁
*
* @param lockKey key
* @param requestId 随机请求id
* @param expireSecond 超时秒
* @param loopTimes 循环次数
* @param sleepInterval 等待间隔(毫秒)
* @return
*/
public static boolean tryGetDistributedLock(String lockKey, String requestId, Integer expireSecond, Long loopTimes, Long sleepInterval) {
DefaultRedisScript<Long> redisScript = new DefaultRedisScript<>(LOCK_SCRIPT_STR, Long.class);
while (loopTimes-- >= 0) {
Object result = redisTemplate.execute(redisScript, Lists.newArrayList(lockKey), requestId, String.valueOf(expireSecond));
if (SUCCESS.equals(result)) {
return true;
}
try {
TimeUnit.MILLISECONDS.sleep(sleepInterval);
} catch (InterruptedException e) {
e.printStackTrace();
}
continue;
}
return false;
}
/**
* 释放锁
*
* @return
*/
public static boolean releaseDistributedLock() {
String callerKey = getCurrentThreadCaller();
String requestId = String.valueOf(Thread.currentThread().getId());
return releaseDistributedLock(callerKey, requestId);
}
/**
* 释放锁
*
* @param lockKey key
* @param requestId 加锁的请求id
* @return
*/
public static boolean releaseDistributedLock(String lockKey, String requestId) {
DefaultRedisScript<Long> redisScript = new DefaultRedisScript<>(UNLOCK_SCRIPT_STR, Long.class);
Object result = redisTemplate.execute(redisScript, Collections.singletonList(lockKey), requestId);
if (SUCCESS.equals(result)) {
return true;
}
return false;
}
private static String getSimpleClassName(String className) {
String[] splits = className.split(PACKAGE_NAME_SPLIT_STR);
return splits[splits.length - 1];
}
/**
* Get caller
*
* @return
*/
private static String getCurrentThreadCaller() {
StackTraceElement stackTraceElement = Thread.currentThread().getStackTrace()[3];
return getSimpleClassName(stackTraceElement.getClassName()) + CLASS_AND_METHOD_CONCAT_STR + stackTraceElement.getMethodName();
}
}