XorList.h 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382
  1. //
  2. // Created by kikab on 06.05.2018.
  3. //
  4. #ifndef XOR_LIST_XORLIST_H
  5. #define XOR_LIST_XORLIST_H
  6. #include <forward_list>
  7. #include <bits/allocator.h>
  8. #include <ext/alloc_traits.h>
  9. #include <vector>
  10. #include <list>
  11. #include <memory>
  12. #include <utility>
  13. template<typename T>
  14. struct XorListNode {
  15. public:
  16. XorListNode() = delete;
  17. XorListNode(const T &val) {
  18. val_ = val;
  19. }
  20. XorListNode(const T &&val) {
  21. val_ = std::forward<XorListNode<T> >(val);
  22. }
  23. XorListNode *right(const XorListNode *left) const {
  24. return reinterpret_cast<XorListNode *>(xor_ptr_ ^ reinterpret_cast<uintptr_t>(left));
  25. }
  26. XorListNode *left(const XorListNode *right) const {
  27. return reinterpret_cast<XorListNode *>(xor_ptr_ ^ reinterpret_cast<uintptr_t>(right));
  28. }
  29. void assign_leftright(const XorListNode *left, const XorListNode *right) {
  30. xor_ptr_ = (reinterpret_cast<uintptr_t>(left) ^ reinterpret_cast<uintptr_t>(right));
  31. }
  32. T val_;
  33. private:
  34. uintptr_t xor_ptr_;
  35. };
  36. template<typename T>
  37. struct FwdXorListIterator {
  38. typedef FwdXorListIterator<T> Self;
  39. typedef XorListNode<T> Node;
  40. typedef ptrdiff_t difference_type;
  41. typedef std::bidirectional_iterator_tag iterator_category;
  42. typedef T value_type;
  43. typedef T *pointer;
  44. typedef T &reference;
  45. FwdXorListIterator() = delete;
  46. FwdXorListIterator(Node *x, Node *prev) : node_(x), left_(prev) {}
  47. // Must downcast from _List_node_base to _List_node to get to value.
  48. reference operator*() const {
  49. return node_->val_;
  50. }
  51. pointer operator->() const {
  52. return &node_->val_;
  53. }
  54. Self &operator++() {
  55. Node *future_left_ = node_;
  56. node_ = node_->right(left_);
  57. left_ = future_left_;
  58. return *this;
  59. }
  60. Self operator++(int) {
  61. Self __tmp = *this;
  62. Node *future_left_ = node_;
  63. node_ = node_->right(left_);
  64. left_ = future_left_;
  65. return __tmp;
  66. }
  67. Self &operator--() {
  68. Node *future_right = node_;
  69. node_ = node_->left(node_->right(left_));
  70. left_ = node_->left(future_right);
  71. return *this;
  72. }
  73. Self operator--(int) {
  74. Self __tmp = *this;
  75. Node *future_right = node_;
  76. node_ = node_->left(node_->right(left_));
  77. left_ = node_->left(future_right);
  78. return __tmp;
  79. }
  80. bool operator==(const Self &x) const {
  81. return left_ == x.left_;
  82. }
  83. bool operator!=(const Self &x) const {
  84. return left_ != x.left_;
  85. }
  86. Node *node_;
  87. Node *left_;
  88. };
  89. template<typename T, typename Allocator = std::allocator<T> >
  90. class XorList {
  91. private:
  92. typedef typename Allocator::template rebind<XorListNode<T>>::other allocator_type;
  93. typedef __gnu_cxx::__alloc_traits<allocator_type> Allocator_traits;
  94. public:
  95. // types:
  96. typedef T value_type;
  97. typedef typename Allocator_traits::pointer pointer;
  98. typedef typename Allocator_traits::const_pointer const_pointer;
  99. typedef value_type& reference;
  100. typedef const value_type& const_reference;
  101. typedef FwdXorListIterator<T> iterator;
  102. typedef std::reverse_iterator<iterator> reverse_iterator;
  103. typedef std::size_t size_type;
  104. typedef std::ptrdiff_t difference_type;
  105. //typedef Allocator allocator_type;
  106. explicit XorList(const Allocator &alloc = Allocator())
  107. : front_(nullptr), back_(nullptr), size_(0), allocator_(allocator_type()) {}
  108. XorList(size_type count, const_reference value = T(), const Allocator &alloc = Allocator())
  109. : front_(nullptr), back_(nullptr), size_(0), allocator_(allocator_type()) {
  110. for (size_type i = 0; i < count; i++)
  111. push_back(value);
  112. }
  113. XorList(const XorList &other) {
  114. *this = other;
  115. }
  116. XorList(const XorList &&other) noexcept {
  117. *this = other;
  118. }
  119. ~XorList() {
  120. XorListNode<T> *xleft = nullptr;
  121. XorListNode<T> *x = front_;
  122. while (x != back_) {
  123. XorListNode<T> *xnext = x->right(xleft);
  124. if (xnext) {
  125. allocator_.destroy(xleft);
  126. allocator_.deallocate(xleft, 1);
  127. }
  128. xleft = x;
  129. x = xnext;
  130. }
  131. if (xleft) {
  132. allocator_.destroy(xleft);
  133. allocator_.deallocate(xleft, 1);
  134. }
  135. if (x) {
  136. allocator_.destroy(x);
  137. allocator_.deallocate(x, 1);
  138. }
  139. }
  140. XorList &operator=(const XorList &other) {
  141. if (this == &other)
  142. return *this;
  143. this->~XorList();
  144. this->front_ = other.front_;
  145. this->back_ = other.back_;
  146. this->size_ = other.size_;
  147. this->allocator_ = other.allocator_;
  148. }
  149. XorList &operator=(const XorList &&other) noexcept {
  150. if (this == &other)
  151. return *this;
  152. this->~XorList();
  153. this->front_ = other.front_;
  154. this->back_ = other.back_;
  155. this->size_ = other.size_;
  156. this->allocator_ = other.allocator_;
  157. }
  158. value_type front() {
  159. return front_->val_;
  160. }
  161. value_type back() {
  162. return back_->val_;
  163. }
  164. iterator begin() {
  165. return iterator(front_, nullptr);
  166. }
  167. iterator end() {
  168. if (!back_)
  169. return iterator(nullptr, nullptr);
  170. return ++iterator(back_, back_->left(nullptr));
  171. }
  172. reverse_iterator rbegin() {
  173. return reverse_iterator(front_, nullptr);
  174. }
  175. reverse_iterator rend() {
  176. return ++reverse_iterator(back_, back_->left(nullptr));
  177. }
  178. size_type size() {
  179. return size_;
  180. }
  181. void push_back(const value_type &val) {
  182. XorListNode<T> *past_back = back_;
  183. back_ = reinterpret_cast<XorListNode<T> *>(allocator_.allocate(1));
  184. allocator_.construct(back_, std::forward<XorListNode<T> >(val));
  185. back_->assign_leftright(past_back, nullptr);
  186. if (past_back)
  187. past_back->assign_leftright(past_back->left(nullptr), back_);
  188. if (!front_)
  189. front_ = back_;
  190. size_++;
  191. }
  192. void push_back(const value_type &&val) {
  193. XorListNode<value_type> *past_back = back_;
  194. back_ = reinterpret_cast<XorListNode<T> *>(allocator_.allocate(1));
  195. allocator_.construct(back_, XorListNode<T>(std::forward<XorListNode<T> >(val)));
  196. back_->assign_leftright(past_back, nullptr);
  197. if (past_back)
  198. past_back->assign_leftright(past_back->left(nullptr), back_);
  199. if (!front_)
  200. front_ = back_;
  201. size_++;
  202. }
  203. void push_front(const value_type &val) {
  204. XorListNode<value_type> *past_front = front_;
  205. front_ = reinterpret_cast<XorListNode<T> *>(allocator_.allocate(1));
  206. allocator_.construct(front_, std::forward<XorListNode<T> >(val));
  207. front_->assign_leftright(nullptr, past_front);
  208. if (past_front)
  209. past_front->assign_leftright(front_, past_front->right(nullptr));
  210. if (!back_)
  211. back_ = front_;
  212. size_++;
  213. }
  214. void push_front(const value_type &&val) {
  215. XorListNode<value_type> *past_front = front_;
  216. front_ = reinterpret_cast<XorListNode<T> *>(allocator_.allocate(1));
  217. allocator_.construct(front_, std::forward<XorListNode<T> >(val));
  218. front_->assign_leftright(nullptr, past_front);
  219. if (past_front)
  220. past_front->assign_leftright(front_, past_front->right(nullptr));
  221. if (!back_)
  222. back_ = front_;
  223. size_++;
  224. }
  225. void pop_back() {
  226. XorListNode<T> *past_back = back_;
  227. back_ = back_->left(nullptr);
  228. back_->assign_leftright(back_->left(past_back), nullptr);
  229. allocator_.destroy(past_back);
  230. allocator_.deallocate(past_back, 1);
  231. }
  232. void pop_front() {
  233. XorListNode<T> *past_front = front_;
  234. front_ = front_->right(nullptr);
  235. front_->assign_leftright(nullptr, front_->right(past_front));
  236. allocator_.destroy(past_front);
  237. allocator_.deallocate(past_front, 1);
  238. }
  239. void insert_before(iterator it, T &val) {
  240. XorListNode<T> *cur = *it;
  241. XorListNode<T> *prev = it.left_;
  242. XorListNode<T> *past = it.node_->right(prev);
  243. XorListNode<T> *new_node = reinterpret_cast<XorListNode<T> *>(allocator_.allocate(1));
  244. allocator_.construct(new_node, std::forward<XorListNode<T> >(val));
  245. new_node->assign_leftright(prev, cur);
  246. if (cur)
  247. cur->assign_leftright(new_node, past);
  248. if (prev)
  249. prev->assign_leftright(prev->left(cur), new_node);
  250. }
  251. void insert_before(iterator it, T &&val) {
  252. XorListNode<T> *cur = *it;
  253. XorListNode<T> *prev = it.left_;
  254. XorListNode<T> *past = it.node_->right(prev);
  255. XorListNode<T> *new_node = allocator_.allocate(1);
  256. allocator_.construct(new_node, std::forward<XorListNode<T> >(val));
  257. new_node->assign_leftright(prev, cur);
  258. if (cur)
  259. cur->assign_leftright(new_node, past);
  260. if (prev)
  261. prev->assign_leftright(prev->left(cur), new_node);
  262. }
  263. void insert_after(iterator it, T &val) {
  264. XorListNode<T> *cur = *it;
  265. XorListNode<T> *prev = it.left_;
  266. XorListNode<T> *past = it.node_->right(prev);
  267. XorListNode<T> *new_node = allocator_.allocate(1);
  268. allocator_.construct(new_node, std::forward<XorListNode<T> >(val));
  269. new_node->assign_leftright(cur, past);
  270. if (cur)
  271. cur->assign_leftright(prev, new_node);
  272. if (past)
  273. past->assign_leftright(new_node, past->right(cur));
  274. }
  275. void insert_after(iterator it, T &&val) {
  276. XorListNode<T> *cur = *it;
  277. XorListNode<T> *prev = it.left_;
  278. XorListNode<T> *past = it.node_->right(prev);
  279. XorListNode<T> *new_node = allocator_.allocate(1);
  280. allocator_.construct(new_node, std::forward<XorListNode<T> >(val));
  281. new_node->assign_leftright(cur, past);
  282. if (cur)
  283. cur->assign_leftright(prev, new_node);
  284. if (past)
  285. past->assign_leftright(new_node, past->right(cur));
  286. }
  287. void erase(iterator it) {
  288. XorListNode<T> *prev = it.left_;
  289. XorListNode<T> *cur = it.operator->();
  290. XorListNode<T> *past = it.node_->right(prev);
  291. if (prev)
  292. prev->assign_leftright(prev->left(cur), past);
  293. if (past)
  294. past->assign_leftright(prev, past->right(cur));
  295. if (cur)
  296. size_--;
  297. allocator_.destroy(cur);
  298. allocator_.deallocate(cur, 1);
  299. }
  300. private:
  301. XorListNode<T> *front_;
  302. XorListNode<T> *back_; // Pointer to the LAST element
  303. size_type size_;
  304. allocator_type allocator_;
  305. };
  306. #endif //XOR_LIST_XORLIST_H