uw.task.util.LeaderVote Maven / Gradle / Ivy
package uw.task.util;
import java.util.Map.Entry;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.TimeUnit;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.BoundValueOperations;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
/**
* 使用Reids的setnx+expire来选举Leader,来在多个运行实例中选举出一个运行全局任务的实例。
*
* @author axeon
*
*/
@Component
public class LeaderVote {
private static final Logger log = LoggerFactory.getLogger(LeaderVote.class);
private static final String REDIS_TAG = "_LEADER_VOTE_";
@Autowired
private StringRedisTemplate template;
/**
* 随机值。
*/
public static final String UUID = java.util.UUID.randomUUID().toString();
/**
* hashmap表。
*/
private ConcurrentHashMap map = new ConcurrentHashMap<>();
/**
* 全局唯一的UUID信息
*
* @return the uUID
*/
public String getUUID() {
return UUID;
}
/**
* 返回当前是否是Leader
*
* @return the isLeader
*/
public boolean isLeader(String name) {
VoteInfo vi = map.get(name);
if (vi == null) {
vi = initVoteInfo(name);
}
if (vi != null)
return vi.isLeader;
else
return false;
}
/**
* 初始化一个voteInfo信息。
*
* @param name
* @return
*/
private synchronized VoteInfo initVoteInfo(String name) {
VoteInfo vi = map.get(name);
if (vi == null) {
try {
vi = new VoteInfo();
vi.bvo = template.boundValueOps(REDIS_TAG + name);
map.put(name, vi);
} catch (Exception e) {
log.error(e.getMessage(), e);
}
}
return vi;
}
/**
* 每20秒检查一次leader状态。
*
* @return true 为leaderStatus
*/
@Scheduled(fixedRate = 20000)
private void checkLeaderStatus() {
try {
for (Entry kv : map.entrySet()) {
if (kv.getValue().bvo == null)
continue;
// 使用setnx来抢leader身份
boolean flag = kv.getValue().bvo.setIfAbsent(UUID);
if (flag) {
// 再次确认身份,并更新expire。
if (UUID.equals(kv.getValue().bvo.get())) {
kv.getValue().bvo.expire(60, TimeUnit.SECONDS);
kv.getValue().isLeader = true;
} else {
kv.getValue().isLeader = false;
}
}
}
} catch (Exception e) {
log.error(e.getMessage(), e);
}
}
/**
* 投票信息
*
* @author axeon
*
*/
private static class VoteInfo {
BoundValueOperations bvo = null;
boolean isLeader = false;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy