PS Lite - 源码解读
PostOffice 类
/**
* \brief 系统的中心。
*/
class Postoffice {
public:
/**
* \brief 返回单例对象。
*/
static Postoffice* Get() {
static Postoffice e; return &e;
} /** \brief 返回持有的 Van 实例。 */
Van* van() { return van_; } /**
* \brief 启动系统。
*
* 本函数将会阻塞,直到所有节点都启动。
* \param argv0 用于日志的程序名。
* \param do_barrier 是否在所有节点都启动完成前阻塞当前线程。
*/
void Start(int customer_id, const char* argv0, const bool do_barrier); /**
* \brief 停止系统。
*
* 所有节点在退出前都应该调用该方法。
* \param do_barrier 是否在所有节点都停止前阻塞当前线程,默认为 true。
*/
void Finalize(const int customer_id, const bool do_barrier = true); /**
* \brief 向本系统添加一个 Customer。线程安全。
*/
void AddCustomer(Customer* customer); /**
* \brief 根据编号移除一个 Customer。线程安全。
*/
void RemoveCustomer(Customer* customer); /**
* \brief 获取给定编号的 Customer。线程安全。
* \param app_id 应用编号。
* \param customer_id Customer 编号。
* \param timeout 以秒位单位的超时。
* \return 如果不存在或者超时,返回 nullptr。
*/
Customer* GetCustomer(int app_id, int customer_id, int timeout = 0) const; /**
* \brief 获取给定编号的节点或节点组。线程安全。
*
* 如果是一个节点组的编号,那么返回所属的全部节点。否则,返回的列表只有一个节点元素。
*/
const std::vector<int>& GetNodeIDs(int node_id) const {
const auto it = node_ids_.find(node_id);
CHECK(it != node_ids_.cend()) << "node " << node_id << " doesn't exist";
return it->second;
} /**
* \brief 返回每个 Server 节点对应的键域。
*/
const std::vector<Range>& GetServerKeyRanges(); /**
* \brief 回调函数的别名。
*/
using Callback = std::function<void()>; /**
* \brief 向系统注册一个回调函数,在 Finalize() 方法完成后被调用。
*
* 以下代码是等价的:
* \code {cpp}
* RegisterExitCallback(cb);
* Finalize();
* \endcode
*
* \code {cpp}
* Finalize();
* cb();
* \endcode
* \param cb 回调函数。
*/
void RegisterExitCallback(const Callback& cb) {
exit_callback_ = cb;
} /**
* \brief 从 Worker 编号映射到节点编号。
* \param rank the worker rank
*/
static inline int WorkerRankToID(int rank) {
return rank * 2 + 9;
} /**
* \brief 从 Server 编号映射到节点编号。
* \param rank the server rank
*/
static inline int ServerRankToID(int rank) {
return rank * 2 + 8;
} /**
* \brief 从节点编号映射到 Server/Worker 编号。
* \param id 节点编号。
*/
static inline int IDtoRank(int id) {
return std::max((id - 8) / 2, 0);
} /** \brief 返回 Worker 数量。 */
int num_workers() const { return num_workers_; } /** \brief 返回 Server 数量。 */
int num_servers() const { return num_servers_; } /** \brief 返回节点在所属分组的编号。
*
* 每个 Worker 都有唯一的编号,范围是 [0, NumWorkers())。Server 也是如此。
* 该方法仅当 Start() 方法被调用后有效。
*/
int my_rank() const { return IDtoRank(van_->my_node().id); } /** \brief 是否是 Worker 节点。 */
int is_worker() const { return is_worker_; } /** \brief 是否是 Server 节点。 */
int is_server() const { return is_server_; } /** \brief 是否是 Scheduler 节点。 */
int is_scheduler() const { return is_scheduler_; } /** \brief 返回日志级别。 */
int verbose() const { return verbose_; } /** \brief 是否是可恢复的节点。 */
bool is_recovery() const { return van_->my_node().is_recovery; } /**
* \brief 屏障。
* \param node_id 要同步的节点组的编号。
*/
void Barrier(int customer_id, int node_group); /**
* \brief 处理控制信息,由持有的 Van 实例调用。
* \param recv 收到的消息。
*/
void Manage(const Message& recv); /**
* \brief 更新心跳记录。
* \param node_id 节点编号。
* \param t 心跳时间戳。
*/
void UpdateHeartbeat(int node_id, time_t t) {
std::lock_guard<std::mutex> lk(heartbeat_mu_);
heartbeats_[node_id] = t;
} /**
* \brief 获取在过去 t 秒内未发送心跳的所有节点。
* \param t 以秒为单位的超时。
*/
std::vector<int> GetDeadNodes(int t = 60);
}
Van 类
/**
* \brief Van 类负责发送消息到远端节点。
*
* 如果环境变量 PS_RESEND 被设置位 1,那么在 PS_RESEND_TIMEOUT 毫秒后没有收到 ACK 消息的情况下,Van 实例会重发消息。
*/
class Van {
public:
/**
* \brief 实例化 Van 类。
* \param type zmq、socket等。
*/
static Van *Create(const std::string &type); /** \brief 空的构造器。使用 Start() 方法来真的启动。
Van() {} /**\brief 空的析构器。使用 Stop() 方法类真的停止。
virtual ~Van() {} /**
* \brief 启动 Van。
*
* 必须先调用该方法,才能调用 Send() 方法。
*
* 该方法初始化到所有节点的连接,启动接受消息的线程。
* 如果收到了控制信息,交给 PostOffice::Manage() 方法处理。否则,交给相应的应用处理。
*/
virtual void Start(int customer_id); /**
* \brief 发送一个消息。线程安全。
* \return 返回发送的字节数。如果发送失败,返回 -1。
*/
int Send(const Message &msg); /**
* \brief 返回所在的节点。
*/
inline const Node &my_node() const {
CHECK(ready_) << "call Start() first";
return my_node_;
} /**
* \brief 停止 Van。
* 停止接受消息的线程。
*/
virtual void Stop(); /**
* \brief 获取下一个可用的时间戳。线程安全。
*/
inline int GetTimestamp() { return timestamp_++; } /**
* \brief 是否可以发送消息。线程安全。
*/
inline bool IsReady() { return ready_; }
}
PS Lite - 源码解读的更多相关文章
- AFNetworking 3.0 源码解读 总结(干货)(上)
养成记笔记的习惯,对于一个软件工程师来说,我觉得很重要.记得在知乎上看到过一个问题,说是人类最大的缺点是什么?我个人觉得记忆算是一个缺点.它就像时间一样,会自己消散. 前言 终于写完了 AFNetwo ...
- AFNetworking 3.0 源码解读(七)之 AFAutoPurgingImageCache
这篇我们就要介绍AFAutoPurgingImageCache这个类了.这个类给了我们临时管理图片内存的能力. 前言 假如说我们要写一个通用的网络框架,除了必备的请求数据的方法外,必须提供一个下载器来 ...
- AFNetworking 3.0 源码解读(一)之 AFNetworkReachabilityManager
做ios开发,AFNetworking 这个网络框架肯定都非常熟悉,也许我们平时只使用了它的部分功能,而且我们对它的实现原理并不是很清楚,就好像总是有一团迷雾在眼前一样. 接下来我们就非常详细的来读一 ...
- AFNetworking 3.0 源码解读(三)之 AFURLRequestSerialization
这篇就讲到了跟请求相关的类了 关于AFNetworking 3.0 源码解读 的文章篇幅都会很长,因为不仅仅要把代码进行详细的的解释,还会大概讲解和代码相关的知识点. 上半篇: URI编码的知识 关于 ...
- AFNetworking 3.0 源码解读(四)之 AFURLResponseSerialization
本篇是AFNetworking 3.0 源码解读的第四篇了. AFNetworking 3.0 源码解读(一)之 AFNetworkReachabilityManager AFNetworking 3 ...
- AFNetworking 3.0 源码解读 总结
终于写完了 AFNetworking 的源码解读.这一过程耗时数天.当我回过头又重头到尾的读了一篇,又有所收获.不禁让我想起了当初上学时的种种情景.我们应该对知识进行反复的记忆和理解.下边是我总结的 ...
- AfNetworking 3.0源码解读
做ios开发,AFNetworking 这个网络框架肯定都非常熟悉,也许我们平时只使用了它的部分功能,而且我们对它的实现原理并不是很清楚,就好像总是有一团迷雾在眼前一样. 接下来我们就非常详细的来读一 ...
- jQuery.Callbacks 源码解读二
一.参数标记 /* * once: 确保回调列表仅只fire一次 * unique: 在执行add操作中,确保回调列表中不存在重复的回调 * stopOnFalse: 当执行回调返回值为false,则 ...
- Jfinal-Plugin源码解读
PS:cnxieyang@163.com/xieyang@e6yun.com 本文就Jfinal-plugin的源码进行分析和解读 Plugin继承及实现关系类图如下,常用的是Iplugin的三个集成 ...
随机推荐
- Shiro-JWT SpringBoot前后端分离权限认证的一种思路
JWT-Shiro 整合 JWT-与Shiro整合进行授权认证的大致思路 图示 大致思路 将登录验证从shiro中分离,自己结合JWT实现 用户登陆后请求认证服务器进行密码等身份信息确认,确认成功后 ...
- Pytest学习笔记3-fixture
前言 个人认为,fixture是pytest最精髓的地方,也是学习pytest必会的知识点. fixture用途 用于执行测试前后的初始化操作,比如打开浏览器.准备测试数据.清除之前的测试数据等等 用 ...
- 关于DWG文件转换成PDF
最近有这样一个需求,客户会提供DWG文件,因为DWG文件是不能直接在网页上显示的,所以必须对他做处理,要求是转换成PDF格式.我查了很久的资料,很多都是基于C#和.NET的方法,而且都是说的很模糊,不 ...
- Golang中的各种时间操作
Golang中的各种时间操作 需求 时间格式的转换比较麻烦,自己写了个工具,可以通过工具中的这些方法相互调用转成自己想要的格式,代码如下,后续有新的函数再添加 实现代码 package utils i ...
- Vue(8)列表渲染v-for
循环 在模板中可以用v-for指令来循环数组,对象等. 循环数组 我们可以用 v-for 指令基于一个数组来渲染一个列表.v-for 指令需要使用 item in items形式的特殊语法,其中 it ...
- 精尽Spring Boot源码分析 - SpringApplication 启动类的启动过程
该系列文章是笔者在学习 Spring Boot 过程中总结下来的,里面涉及到相关源码,可能对读者不太友好,请结合我的源码注释 Spring Boot 源码分析 GitHub 地址 进行阅读 Sprin ...
- 上海某大公司:你是了解Redis对吧?
<对线面试官>系列目前已经连载26篇啦!有深度风趣的系列! [对线面试官]Java注解 [对线面试官]Java泛型 [对线面试官] Java NIO [对线面试官]Java反射 & ...
- 9.5、zabbix高级操作(2)
4.zabbix的分布式监控: 使用zabbix-proxy主动方式(被动也可),使用zabbix-agent的主动方式(被动也可): Zabbix Server <- Zabbix Proxy ...
- 浅读tomcat架构设计之tomcat容器Container(3)
浅读tomcat架构设计和tomcat启动过程(1) https://www.cnblogs.com/piaomiaohongchen/p/14977272.html 浅读tomcat架构设计之tom ...
- Linux:VMware配置NAT网络IP
设置虚拟机网络配置 在目标虚拟机下右键, 选择"设置", 打开"虚拟机设置"对话框, 再选择"网络适配器"使用NAT模式的, 如下图所示: ...