首页 技术 正文
技术 2022年11月17日
0 收藏 926 点赞 3,821 浏览 3531 个字

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_; }
}
相关推荐
python开发_常用的python模块及安装方法
adodb:我们领导推荐的数据库连接组件bsddb3:BerkeleyDB的连接组件Cheetah-1.0:我比较喜欢这个版本的cheeta…
日期:2022-11-24 点赞:878 阅读:9,499
Educational Codeforces Round 11 C. Hard Process 二分
C. Hard Process题目连接:http://www.codeforces.com/contest/660/problem/CDes…
日期:2022-11-24 点赞:807 阅读:5,913
下载Ubuntn 17.04 内核源代码
zengkefu@server1:/usr/src$ uname -aLinux server1 4.10.0-19-generic #21…
日期:2022-11-24 点赞:569 阅读:6,746
可用Active Desktop Calendar V7.86 注册码序列号
可用Active Desktop Calendar V7.86 注册码序列号Name: www.greendown.cn Code: &nb…
日期:2022-11-24 点赞:733 阅读:6,503
Android调用系统相机、自定义相机、处理大图片
Android调用系统相机和自定义相机实例本博文主要是介绍了android上使用相机进行拍照并显示的两种方式,并且由于涉及到要把拍到的照片显…
日期:2022-11-24 点赞:512 阅读:8,142
Struts的使用
一、Struts2的获取  Struts的官方网站为:http://struts.apache.org/  下载完Struts2的jar包,…
日期:2022-11-24 点赞:671 阅读:5,305