qxLib
unique_objects_pool.h
Go to the documentation of this file.
1 /**
2 
3  @file unique_objects_pool.h
4  @author Khrapov
5  @date 11.02.2023
6  @copyright © Nick Khrapov, 2023. All right reserved.
7 
8 **/
9 #pragma once
10 
11 #include <qx/category.h>
12 #include <qx/internal/perf_scope.h>
14 #include <qx/typedefs.h>
15 
16 QX_PUSH_SUPPRESS_ALL_WARNINGS();
17 #include <boost/multi_index/hashed_index.hpp>
18 #include <boost/multi_index/member.hpp>
19 #include <boost/multi_index_container.hpp>
20 QX_POP_SUPPRESS_WARNINGS();
21 
22 #include <algorithm>
23 #include <mutex>
24 #include <shared_mutex>
25 
26 QX_DEFINE_CATEGORY(CatUniqueObjectsPool, qx::color::honeydew());
27 
28 namespace qx
29 {
30 
31 /**
32 
33  @class unique_objects_pool
34  @brief Class stores unique objects and allows access to them through tokens
35  @details Class is thread safe
36  @tparam T - object type
37  @author Khrapov
38  @date 11.02.2023
39 
40 **/
41 template<class T>
43 {
44  struct data
45  {
46  T value;
47  u64 nId = 0;
48  u64 nCounter = 0;
49  };
50 
51  struct value_tag;
52  struct id_tag;
53 
54  using values_set = boost::multi_index_container<
55  data,
56  boost::multi_index::indexed_by<
57  boost::multi_index::
58  hashed_unique<boost::multi_index::tag<value_tag>, boost::multi_index::member<data, T, &data::value> >,
59  boost::multi_index::
60  hashed_unique<boost::multi_index::tag<id_tag>, boost::multi_index::member<data, u64, &data::nId> > > >;
61 
62  using data_by_value = typename boost::multi_index::index<values_set, value_tag>::type;
63  using data_by_id = typename boost::multi_index::index<values_set, id_tag>::type;
64 
65 public:
66  /**
67 
68  @class token
69  @brief The token is used for distributed access to a unique object in the pool
70  @details Class is thread safe
71  @author Khrapov
72  @date 11.02.2023
73 
74  **/
75  class token
76  {
77  friend unique_objects_pool;
78 
79  static constexpr u64 kInvalidId = static_cast<u64>(-1);
80 
81  public:
82  token() noexcept = default;
83  token(const token& otherToken) noexcept;
84  token(token&& otherToken) noexcept;
85 
86  token& operator=(const token& otherToken) noexcept;
87  token& operator=(token&& otherToken) noexcept;
88 
89  ~token() noexcept;
90 
91  /**
92  @brief Is this token have been set (not default)
93  @warning This method does not check if the pool of this object is alive
94  @retval - true if this object is valid
95  **/
96  bool is_valid() const noexcept;
97 
98  /**
99  @brief Same as is_valid()
100  @retval - true if this object is valid
101  **/
102  explicit operator bool() const noexcept;
103 
104  const T& operator*() const noexcept;
105  const T* operator->() const noexcept;
106  bool operator==(const token& other) const noexcept;
107 
108  private:
109  /**
110  @brief token object constructor
111  @param nId - object id
112  @param pPool - pool pointer
113  **/
114  token(u64 nId, unique_objects_pool* pPool) noexcept;
115 
116  private:
117  u64 m_nId = kInvalidId;
118  unique_objects_pool* m_pPool = nullptr;
119  };
120 
121 public:
122  /**
123  @brief unique_objects_pool object constructor
124  @param bAutoCleanup - if true, objects will be deleted when usages count reaches 0, otherwise you should use shrink()
125  **/
126  unique_objects_pool(bool bAutoCleanup = true) noexcept;
127 
128  /**
129  @brief Get or create object
130  @tparam U - value type
131  @param value - object to search and to construct if wasn't able to find
132  @retval - token to existing or constructed object
133  **/
134  template<class U>
135  token get_or_create(U&& value) noexcept;
136 
137  /**
138  @brief Remove unused (with counter = 0) objects
139  **/
140  void shrink();
141 
142  /**
143  @brief Get number of objects stored
144  @retval - number of objects stored
145  **/
146  size_t size() const;
147 
148  /**
149  @brief Is pool empty
150  @retval - true if pool is empty
151  **/
152  bool empty() const;
153 
154 private:
155  /**
156  @brief Increase the usage counter of the object with the specified id
157  @param nId - object id
158  **/
159  void increase_counter(u64 nId) noexcept;
160 
161  /**
162  @brief Decrease the usage counter of the object with the specified id
163  @param nId - object id
164  **/
165  void decrease_counter(u64 nId) noexcept;
166 
167  /**
168  @brief Get value of the object with the specified id
169  @param nId - object id
170  @retval - object value
171  **/
172  const T& get_value(u64 nId) noexcept;
173 
174 private:
175  const bool m_bAutoShrink = true;
176 
177  QX_PERF_SHARED_MUTEX(m_UniqueObjectsPoolMutex);
178  values_set m_Pool;
179  u64 m_nCurrentId = 1;
180 };
181 
182 } // namespace qx
183 
184 #include <qx/containers/unique_objects_pool.inl>
#define QX_DEFINE_CATEGORY(name,...)
Define a category.
Definition: category.h:21
The token is used for distributed access to a unique object in the pool.
bool is_valid() const noexcept
Is this token have been set (not default)
Class stores unique objects and allows access to them through tokens.
unique_objects_pool(bool bAutoCleanup=true) noexcept
unique_objects_pool object constructor
token get_or_create(U &&value) noexcept
Get or create object.
bool empty() const
Is pool empty.
void shrink()
Remove unused (with counter = 0) objects.
size_t size() const
Get number of objects stored.
Contains perf scope macros for profiler (for internal usage only, but user may override them)
uint64_t u64
Definition: typedefs.h:27