// 从磁盘加载Datum和term数据进行数据恢复
RaftStore.load();
Loggers.RAFT.info(“cache loaded, peer count: {}, datum count: {}, current term: {}”,
peers.size(), datums.size(), peers.getTerm());
while (true) {
if (notifier.tasks.size() <= 0) {
break;
}
Thread.sleep(1000L);
System.out.println(notifier.tasks.size());
}
Loggers.RAFT.info(“finish to load data from disk, cost: {} ms.”, (System.currentTimeMillis() - start));
GlobalExecutor.register(new MasterElection()); // Leader选举
GlobalExecutor.register1(new HeartBeat()); // Raft心跳
GlobalExecutor.register(new AddressServerUpdater(), GlobalExecutor.ADDRESS_SERVER_UPDATE_INTERVAL_MS);
if (peers.size() > 0) {
if (lock.tryLock(INIT_LOCK_TIME_SECONDS, TimeUnit.SECONDS)) {
initialized = true;
lock.unlock();
}
} else {
throw new Exception(“peers is empty.”);
}
Loggers.RAFT.info(“timer started: leader timeout ms: {}, heart-beat timeout ms: {}”,
GlobalExecutor.LEADER_TIMEOUT_MS, GlobalExecutor.HEARTBEAT_INTERVAL_MS);
}
在init方法主要做了如下几件事:
-
获取Raft集群节点 peers.add(NamingProxy.getServers());
-
Raft集群数据恢复 RaftStore.load();
-
Raft选举 GlobalExecutor.register(new MasterElection());
-
Raft心跳 GlobalExecutor.register(new HeartBeat());
-
Raft发布内容
-
Raft保证内容一致性
[](()选举流程
其中,raft集群内部节点间是通过暴露的Restful接口,代码在 RaftController 中。RaftController控制器是raft集群内部节点间通信使用的,具体的信息如下
POST 《大厂前端面试题解析+Web核心总结学习笔记+企业项目实战源码+最新高清讲解视频》无偿开源 徽信搜索公众号【编程进阶路】 HTTP://{ip:port}/v1/ns/raft/vote : 进行投票请求
POST HTTP://{ip:port}/v1/ns/raft/beat : Leader向Follower发送心跳信息
GET HTTP://{ip:port}/v1/ns/raft/peer : 获取该节点的RaftPeer信息
PUT HTTP://{ip:port}/v1/ns/raft/datum/reload : 重新加载某日志信息
POST HTTP://{ip:port}/v1/ns/raft/datum : Leader接收传来的数据并存入
DELETE HTTP://{ip:port}/v1/ns/raft/datum : Leader接收传来的数据删除操作
GET HTTP://{ip:port}/v1/ns/raft/datum : 获取该节点存储的数据信息
GET HTTP://{ip:port}/v1/ns/raft/state : 获取该节点的状态信息{UP or DOWN}
POST HTTP://{ip:port}/v1/ns/raft/datum/commit : Follower节点接收Leader传来得到数据存入操作
DELETE HTTP://{ip:port}/v1/ns/raft/datum : Follower节点接收Leader传来的数据删除操作
GET HTTP://{ip:port}/v1/ns/raft/leader : 获取当前集群的Leader节点信息
GET HTTP://{ip:port}/v1/ns/raft/listeners : 获取当前Raft集群的所有事件监听者
RaftPeerSet
[](()心跳机制
Raft中使用心跳机制来触发leader选举。心跳定时任务是在GlobalExecutor 中,通过 GlobalExecutor.register(new HeartBeat())注册心跳定时任务,具体操作包括:
-
重置Leader节点的heart timeout、election timeout;
-
sendBeat()发送心跳包
public class HeartBeat implements Runnable {
@Override
public void run() {
try {
if (!peers.isReady()) {
return;
}
RaftPeer local = peers.local();
local.heartbeatDueMs -= GlobalExecutor.TICK_PERIOD_MS;
if (local.heartbeatDueMs > 0) {
return;
}
local.resetHeartbeatDue();
sendBeat();
} catch (Exception e) {
Loggers.RAFT.warn(“[RAFT] error while sending beat {}”, e);
}
}
}
简单说明了下Nacos中的Raft一致性实现,更详细的流程,可以下载源码,查看 RaftCore 进行了解。源码可以通过以下地址检出:
git clone https://github.com/alibaba/nacos.git
[](()最后
分享一套我整理的面试干货,这份文档结合了我多年的面试官经验,站在面试官的角度来告诉你,面试官提的那些问题他最想听到你给他的回答是什么,分享出来帮助那些对前途感到迷茫的朋友。
[](()面试经验技巧篇
-
经验技巧1 如何巧妙地回答面试官的问题
-
经验技巧2 如何回答技术性的问题
-
经验技巧3 如何回答非技术性问题
-
经验技巧4 如何回答快速估算类问题
-
经验技巧5 如何回答算法设计问题
-
经验技巧6 如何回答系统设计题
-
经验技巧7 如何解决求职中的时间冲突问题
-
经验技巧8 如果面试问题曾经遇见过,是否要告知面试官
-
经验技巧9 在被企业拒绝后是否可以再申请
-
经验技巧10 如何应对自己不会回答的问题
-
经验技巧11 如何应对面试官的“激将法”语言
-
经验技巧12 如何处理与面试官持不同观点这个问题
-
经验技巧13 什么是职场暗语
[](()面试真题篇
- 真题详解1 某知名互联网下载服务提供商软件工程师笔试题