2018-12-28 21:46:01 +08:00
|
|
|
|
#pragma once
|
|
|
|
|
#include "fiber_cpp_define.hpp"
|
2019-01-14 14:34:14 +08:00
|
|
|
|
#include <string>
|
2018-12-28 21:46:01 +08:00
|
|
|
|
|
|
|
|
|
namespace acl {
|
|
|
|
|
|
2019-01-14 14:34:14 +08:00
|
|
|
|
class keeper_waiter;
|
2018-12-28 21:46:01 +08:00
|
|
|
|
class socket_stream;
|
2019-01-14 14:34:14 +08:00
|
|
|
|
class thread_mutex;
|
2018-12-28 21:46:01 +08:00
|
|
|
|
|
2019-01-15 11:57:19 +08:00
|
|
|
|
/**
|
|
|
|
|
* 独立线程用于预先与服务器创建空闲连接,客户端可以直接从该连接池中获取新连接,
|
|
|
|
|
* 这对于 ping rtt 较长(如:10ms 以上)比较有价值,可以有效地减少因网络 rtt
|
|
|
|
|
* 造成的连接时间损耗
|
|
|
|
|
*/
|
2018-12-28 21:46:01 +08:00
|
|
|
|
class tcp_keeper : public thread
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
tcp_keeper(void);
|
|
|
|
|
~tcp_keeper(void);
|
|
|
|
|
|
2019-01-15 11:57:19 +08:00
|
|
|
|
/**
|
|
|
|
|
* 设置建立网络连接的超时时间(秒)
|
|
|
|
|
* @param n {int}
|
|
|
|
|
* @return {tcp_keeper&}
|
|
|
|
|
*/
|
2018-12-29 15:42:03 +08:00
|
|
|
|
tcp_keeper& set_conn_timeout(int n);
|
2019-01-15 11:57:19 +08:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 设置网络套接字 IO 读写超时时间(秒)
|
|
|
|
|
* @param n {int}
|
|
|
|
|
* @return {tcp_keeper&}
|
|
|
|
|
*/
|
2018-12-29 15:42:03 +08:00
|
|
|
|
tcp_keeper& set_rw_timeout(int n);
|
2019-01-15 11:57:19 +08:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 设置连接池中空闲连接的最小连接数
|
|
|
|
|
* @param n {int}
|
|
|
|
|
* @return {tcp_keeper&}
|
|
|
|
|
*/
|
|
|
|
|
tcp_keeper& set_conn_min(int n);
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 设置连接池中空闲连接的最大连接数
|
|
|
|
|
* @param n {int}
|
|
|
|
|
* @return {tcp_keeper&}
|
|
|
|
|
*/
|
2018-12-29 15:42:03 +08:00
|
|
|
|
tcp_keeper& set_conn_max(int n);
|
2019-01-15 11:57:19 +08:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 设置网络连接的空闲时间(秒),空闲时间超过此值时连接将被关闭
|
|
|
|
|
* @param ttl {int}
|
|
|
|
|
* @return {tcp_keeper&}
|
|
|
|
|
*/
|
2018-12-29 15:42:03 +08:00
|
|
|
|
tcp_keeper& set_conn_ttl(int ttl);
|
2019-01-15 11:57:19 +08:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 设置每个连接池的空闲时间(秒),即当该连接池的空闲时间超过此值时
|
|
|
|
|
* 将被释放,从而便于系统回收内存资源
|
|
|
|
|
* @param ttl {int}
|
|
|
|
|
* @return {tcp_keeper&}
|
|
|
|
|
*/
|
2018-12-29 15:42:03 +08:00
|
|
|
|
tcp_keeper& set_pool_ttl(int ttl);
|
|
|
|
|
|
2019-01-15 11:57:19 +08:00
|
|
|
|
/**
|
|
|
|
|
* 设置 rtt 阀值(秒),当网络连接时间超过此值时才会启用从连接池提取
|
|
|
|
|
* 连接方式,如果网络连接时间小于此值,则直接连接服务器
|
|
|
|
|
* @param rtt {double}
|
|
|
|
|
* @return {tcp_keeper&}
|
|
|
|
|
*/
|
2019-01-14 14:34:14 +08:00
|
|
|
|
tcp_keeper& set_rtt_min(double rtt);
|
|
|
|
|
|
2019-01-15 11:57:19 +08:00
|
|
|
|
/**
|
|
|
|
|
* 从 tcp_keeper 对象中提取对应地址的网络连接对接
|
|
|
|
|
* @param addr {const char*} 服务器地址,格式:ip:port
|
|
|
|
|
* @param hit {bool*} 非空时,将存放该连接是否在连接池的空闲连接中命中
|
2019-04-30 17:54:08 +08:00
|
|
|
|
* @param sync {bool} 是否采用直连模式,如果采用直连模式,则内部不会
|
|
|
|
|
* 针对该地址预创连接池
|
2019-01-15 11:57:19 +08:00
|
|
|
|
* @return {socket_stream*} 返回 NULL 表示连接失败
|
|
|
|
|
*/
|
2019-04-30 17:54:08 +08:00
|
|
|
|
socket_stream* peek(const char* addr, bool* hit = NULL,
|
|
|
|
|
bool sync = false);
|
2018-12-28 21:46:01 +08:00
|
|
|
|
|
2019-01-15 11:57:19 +08:00
|
|
|
|
/**
|
|
|
|
|
* 停止 tcp_keeper 线程运行
|
|
|
|
|
*/
|
2018-12-28 21:46:01 +08:00
|
|
|
|
void stop(void);
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
// @override
|
|
|
|
|
void* run(void);
|
|
|
|
|
|
|
|
|
|
private:
|
2019-01-14 14:34:14 +08:00
|
|
|
|
double rtt_min_;
|
|
|
|
|
keeper_waiter* waiter_;
|
|
|
|
|
std::map<std::string, double> addrs_;
|
|
|
|
|
thread_mutex* lock_;
|
|
|
|
|
|
|
|
|
|
bool direct(const char* addr, bool& found);
|
|
|
|
|
void remove(const char* addr);
|
|
|
|
|
void update(const char* addr, double cost);
|
2018-12-28 21:46:01 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
} // namespace acl
|