#pragma once #include #include #include #include #include /// This type specifies the possible behaviors of an object pool allocator. enum class PoolMode { /// Get exactly one object from a given pool. GET_ONE = 0, /// Get a number of objects from a given pool, this number being /// read from a configuration parameter. GET_MANY, /// Get all the objects from a given pool. GET_ALL }; /** Класс, от которого можно унаследоваться и получить пул чего-нибудь. Используется для пулов соединений с БД. * Наследник должен предоставить метод для создания нового объекта для помещения в пул. */ template class PoolBase : private boost::noncopyable { public: using Object = TObject; using ObjectPtr = std::shared_ptr; using Ptr = std::shared_ptr>; private: /** Объект с флагом, используется ли он сейчас. */ struct PooledObject { PooledObject(std::condition_variable & available_, ObjectPtr object_) : object(object_), available(available_) { } ObjectPtr object; bool in_use = false; std::condition_variable & available; }; using Objects = std::vector>; /** Помощник, который устанавливает флаг использования объекта, а в деструкторе - снимает, * а также уведомляет о событии с помощью condvar-а. */ struct PoolEntryHelper { PoolEntryHelper(PooledObject & data_) : data(data_) { data.in_use = true; } ~PoolEntryHelper() { data.in_use = false; data.available.notify_one(); } PooledObject & data; }; public: /** То, что выдаётся пользователю. */ class Entry { public: friend class PoolBase; Entry() {} /// Для отложенной инициализации. /** Объект Entry защищает ресурс от использования другим потоком. * Следующие методы запрещены для rvalue, чтобы нельзя было написать подобное * * auto q = pool.Get()->query("SELECT .."); // Упс, после этой строчки Entry уничтожился * q.execute(); // Кто-то еще может использовать этот Connection */ Object * operator->() && = delete; const Object * operator->() const && = delete; Object & operator*() && = delete; const Object & operator*() const && = delete; Object * operator->() & { return &*data->data.object; } const Object * operator->() const & { return &*data->data.object; } Object & operator*() & { return *data->data.object; } const Object & operator*() const & { return *data->data.object; } bool isNull() const { return data.isNull(); } private: std::shared_ptr data; Entry(PooledObject & object) : data(std::make_shared(object)) {} }; virtual ~PoolBase() {} /** Выделяет объект для работы. При timeout < 0 таймаут бесконечный. */ Entry get(Poco::Timespan::TimeDiff timeout) { std::unique_lock lock(mutex); while (true) { for (auto & item : items) if (!item->in_use) return Entry(*item); if (items.size() < max_items) { ObjectPtr object = allocObject(); items.emplace_back(std::make_shared(available, object)); return Entry(*items.back()); } LOG_INFO(log, "No free connections in pool. Waiting."); if (timeout < 0) available.wait(lock); else available.wait_for(lock, std::chrono::microseconds(timeout)); } } void reserve(size_t count) { std::lock_guard lock(mutex); while (items.size() < count) items.emplace_back(std::make_shared(available, allocObject())); } private: /** Максимальный размер пула. */ unsigned max_items; /** Пул. */ Objects items; /** Блокировка для доступа к пулу. */ std::mutex mutex; std::condition_variable available; protected: Logger * log; PoolBase(unsigned max_items_, Logger * log_) : max_items(max_items_), log(log_) { items.reserve(max_items); } /** Создает новый объект для помещения в пул. */ virtual ObjectPtr allocObject() = 0; };