qxLib
generic_span.h
Go to the documentation of this file.
1 /**
2 
3  @file generic_span.h
4  @author Khrapov
5  @date 20.11.2022
6  @copyright © Nick Khrapov, 2022. All right reserved.
7 
8 **/
9 #pragma once
10 
11 #include <qx/meta/qualifiers.h>
12 
13 #include <functional>
14 
15 namespace qx
16 {
17 
19 {
20  template<class... args_t>
21  using function_type = std::function<args_t...>;
22 };
23 
24 /**
25 
26  @class generic_span
27  @brief Same as std::span, but may be used with containers with forward iterators
28  @details The class can be used to use any container to iterate over its elements
29  Useful when used in virtual methods
30  Does not own the container
31  The container should not be changed until the end of the iteration
32  May be much slower then std::span when the size of sbo is smaller then sizeof 2 iterators of this container
33  (libcpp: 16, ligcxx: 16, MSVC: 56)
34  @tparam T - container value type
35  @author Khrapov
36  @date 20.11.2022
37 
38 **/
39 template<class T, class traits_t = default_generic_span_traits>
41 {
42  template<class container_t>
43  using container_value = std::remove_reference_t<decltype(*std::declval<container_t>().begin())>;
44 
45  template<class target_t>
46  using qualified_type = copy_qualifiers_t<std::remove_reference_t<std::remove_pointer_t<T>>, target_t>;
47 
48 public:
49  using traits = traits_t;
50  template<class... args_t>
51  using function_type = typename traits_t::template function_type<args_t...>;
52  using generator_type = function_type<T*()>;
53 
54  using element_type = T;
55  using value_type = std::remove_cv_t<T>;
56  using pointer = T*;
57  using reference = T&;
58 
59  class iterator
60  {
61  public:
62  using value_type = T;
63  using pointer = T*;
64  using reference = T&;
65  using difference_type = std::ptrdiff_t;
66  using size_type = size_t;
67  using iterator_category = std::forward_iterator_tag;
68  using iterator_concept = std::forward_iterator_tag;
69 
70  public:
71  constexpr iterator() noexcept = default;
72  constexpr iterator(const iterator&) noexcept = default;
73  constexpr iterator(generator_type generator, size_type nIndex) noexcept;
74  [[nodiscard]] constexpr reference operator*() const noexcept;
75  [[nodiscard]] constexpr pointer operator->() const noexcept;
76  constexpr iterator& operator++() noexcept;
77  [[nodiscard]] constexpr iterator operator++(int) noexcept;
78  constexpr bool operator!=(const iterator& r) const noexcept;
79  constexpr bool operator==(const iterator& r) const noexcept;
80  constexpr operator void*() const noexcept;
81 
82  private:
83  generator_type m_Generator;
84  size_type m_nIndex = 0;
85  };
86 
87 public:
88  generic_span() noexcept = default;
89  generic_span(generic_span&&) noexcept = default;
90  generic_span(const generic_span&) noexcept = default;
91 
92  /**
93  @brief generic_span object constructor
94  @tparam container_t - any container, which adapter satisfies forward iterator concept
95  @param container - container instance. container must not be edited until adapter death
96  **/
97  template<class container_t>
98  generic_span(container_t& container) noexcept;
99 
100  /**
101  @brief generic_span object constructor
102  @tparam container_t - any container, which adapter satisfies forward iterator concept
103  @param container - container instance. container must not be edited until adapter death
104  @param valueAdapter - function which transforms Container::value_type to T*
105  **/
106  template<class container_t>
107  generic_span(
108  container_t& container,
109  function_type<reference(qualified_type<container_value<container_t>>&)> valueAdapter) noexcept;
110 
111  /**
112  @brief operator=
113  @tparam container_t - any container, which adapter satisfies forward iterator concept
114  @param container - container instance. container must not be edited until adapter death
115  @retval - this object reference
116  **/
117  template<class container_t>
118  generic_span& operator=(container_t& container) noexcept;
119 
120  generic_span& operator=(generic_span&&) noexcept = default;
121  generic_span& operator=(const generic_span&) noexcept = default;
122 
123  /**
124  @brief Check if span is empty
125  @retval - true if empty
126  **/
127  bool empty() const noexcept;
128 
129  /**
130  @brief Return iterator to beginning
131  @retval - iterator to beginning
132  **/
133  iterator begin() const noexcept;
134 
135  /**
136  @brief Return iterator to end
137  @retval - iterator to end
138  **/
139  iterator end() const noexcept;
140 
141  /**
142  @brief Return const iterator to beginning
143  @retval - const iterator to beginning
144  **/
145  iterator cbegin() const noexcept;
146 
147  /**
148  @brief Return const iterator to end
149  @retval - const iterator to end
150  **/
151  iterator cend() const noexcept;
152 
153 private:
154  /**
155  @brief Create initial generator
156  @tparam container_t - any container, which adapter satisfies forward iterator concept
157  @tparam adapter_t - adapter callable type
158  @param container - container instance. container must not be edited until adapter death
159  @param adapter - function which transforms Container::value_type to T*
160  @retval - values generator
161  **/
162  template<class container_t, class adapter_t>
163  static generator_type create_initial_generator(container_t&& container, adapter_t adapter) noexcept;
164 
165 private:
166  generator_type m_InitialGenerator;
167 };
168 
169 } // namespace qx
170 
171 #include <qx/patterns/generic_span.inl>
Same as std::span, but may be used with containers with forward iterators.
Definition: generic_span.h:41
iterator begin() const noexcept
Return iterator to beginning.