mirror of
https://gitee.com/fasiondog/hikyuu.git
synced 2024-12-05 05:17:47 +08:00
124 lines
3.1 KiB
C++
124 lines
3.1 KiB
C++
/*
|
|
* ConnectPool.h
|
|
*
|
|
* Copyright (c) 2019, hikyuu.org
|
|
*
|
|
* Created on: 2019-8-5
|
|
* Author: fasiondog
|
|
*/
|
|
#pragma once
|
|
#ifndef HIKYUU_UTILITIES_CONNECTPOOL_H
|
|
#define HIKYUU_UTILITIES_CONNECTPOOL_H
|
|
|
|
#include <thread>
|
|
#include <mutex>
|
|
#include <queue>
|
|
#include "Parameter.h"
|
|
|
|
namespace hku {
|
|
|
|
/**
|
|
* 通用连接池
|
|
* @ingroup Utilities
|
|
*/
|
|
template <typename ConnectType>
|
|
class ConnectPool {
|
|
public:
|
|
ConnectPool() = delete;
|
|
ConnectPool(const ConnectPool &) = delete;
|
|
ConnectPool &operator=(const ConnectPool &) = delete;
|
|
|
|
/**
|
|
* 构造函数
|
|
* @param param 连接参数
|
|
* @param maxConnect 允许的最大连接数,为 0 表示不限制
|
|
* @param maxIdleConnect 运行的最大空闲连接数,为 0 表示不限制
|
|
*/
|
|
explicit ConnectPool(const Parameter ¶m, size_t maxConnect = 0, size_t maxIdleConnect = 0)
|
|
: m_maxConnectSize(0),
|
|
m_maxIdelSize(maxIdleConnect),
|
|
m_count(0),
|
|
m_param(param),
|
|
m_closer(this) {}
|
|
|
|
/**
|
|
* 析构函数,释放所有缓存的连接
|
|
*/
|
|
virtual ~ConnectPool() {
|
|
while (!m_connectList.empty()) {
|
|
ConnectType *p = m_connectList.front();
|
|
m_connectList.pop();
|
|
if (p) {
|
|
delete p;
|
|
}
|
|
}
|
|
}
|
|
|
|
/** 连接实例指针类型 */
|
|
typedef std::shared_ptr<ConnectType> ConnectPtr;
|
|
|
|
/** 获取可用连接,如超出允许的最大连接数将返回空指针 */
|
|
ConnectPtr getConnect() noexcept {
|
|
std::lock_guard<std::mutex> lock(m_mutex);
|
|
if (m_maxConnectSize > 0 && m_count > m_maxConnectSize) {
|
|
return ConnectPtr();
|
|
}
|
|
if (m_connectList.empty()) {
|
|
m_count++;
|
|
return ConnectPtr(new ConnectType(m_param), m_closer);
|
|
}
|
|
ConnectType *p = m_connectList.front();
|
|
m_connectList.pop();
|
|
return ConnectPtr(p, m_closer);
|
|
}
|
|
|
|
/** 当前活动的连接数 */
|
|
size_t count() const {
|
|
return m_count;
|
|
}
|
|
|
|
private:
|
|
/** 归还至连接池 */
|
|
void returnDriver(ConnectType *p) {
|
|
std::lock_guard<std::mutex> lock(m_mutex);
|
|
if (p) {
|
|
if (0 == m_maxIdelSize || m_connectList.size() < m_maxIdelSize) {
|
|
m_connectList.push(p);
|
|
} else {
|
|
delete p;
|
|
p = nullptr;
|
|
m_count--;
|
|
}
|
|
} else {
|
|
m_count--;
|
|
HKU_WARN("Trying to return an empty pointer!");
|
|
}
|
|
}
|
|
|
|
private:
|
|
size_t m_maxConnectSize; //允许的最大连接数
|
|
size_t m_maxIdelSize; //允许的最大空闲连接数
|
|
size_t m_count; //当前活动的连接数
|
|
Parameter m_param;
|
|
std::mutex m_mutex;
|
|
std::queue<ConnectType *> m_connectList;
|
|
|
|
class ConnectCloser {
|
|
public:
|
|
explicit ConnectCloser(ConnectPool *pool) : m_pool(pool) {}
|
|
void operator()(ConnectType *conn) {
|
|
if (m_pool && conn) {
|
|
m_pool->returnDriver(conn);
|
|
}
|
|
}
|
|
|
|
private:
|
|
ConnectPool *m_pool;
|
|
};
|
|
|
|
ConnectCloser m_closer;
|
|
};
|
|
|
|
} // namespace hku
|
|
|
|
#endif /* HIKYUU_UTILITIES_CONNECTPOOL_H */ |