123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382 |
- //
- // Created by kikab on 06.05.2018.
- //
- #ifndef XOR_LIST_XORLIST_H
- #define XOR_LIST_XORLIST_H
- #include <forward_list>
- #include <bits/allocator.h>
- #include <ext/alloc_traits.h>
- #include <vector>
- #include <list>
- #include <memory>
- #include <utility>
- template<typename T>
- struct XorListNode {
- public:
- XorListNode() = delete;
- XorListNode(const T &val) {
- val_ = val;
- }
- XorListNode(const T &&val) {
- val_ = std::forward<XorListNode<T> >(val);
- }
- XorListNode *right(const XorListNode *left) const {
- return reinterpret_cast<XorListNode *>(xor_ptr_ ^ reinterpret_cast<uintptr_t>(left));
- }
- XorListNode *left(const XorListNode *right) const {
- return reinterpret_cast<XorListNode *>(xor_ptr_ ^ reinterpret_cast<uintptr_t>(right));
- }
- void assign_leftright(const XorListNode *left, const XorListNode *right) {
- xor_ptr_ = (reinterpret_cast<uintptr_t>(left) ^ reinterpret_cast<uintptr_t>(right));
- }
- T val_;
- private:
- uintptr_t xor_ptr_;
- };
- template<typename T>
- struct FwdXorListIterator {
- typedef FwdXorListIterator<T> Self;
- typedef XorListNode<T> Node;
- typedef ptrdiff_t difference_type;
- typedef std::bidirectional_iterator_tag iterator_category;
- typedef T value_type;
- typedef T *pointer;
- typedef T &reference;
- FwdXorListIterator() = delete;
- FwdXorListIterator(Node *x, Node *prev) : node_(x), left_(prev) {}
- // Must downcast from _List_node_base to _List_node to get to value.
- reference operator*() const {
- return node_->val_;
- }
- pointer operator->() const {
- return &node_->val_;
- }
- Self &operator++() {
- Node *future_left_ = node_;
- node_ = node_->right(left_);
- left_ = future_left_;
- return *this;
- }
- Self operator++(int) {
- Self __tmp = *this;
- Node *future_left_ = node_;
- node_ = node_->right(left_);
- left_ = future_left_;
- return __tmp;
- }
- Self &operator--() {
- Node *future_right = node_;
- node_ = node_->left(node_->right(left_));
- left_ = node_->left(future_right);
- return *this;
- }
- Self operator--(int) {
- Self __tmp = *this;
- Node *future_right = node_;
- node_ = node_->left(node_->right(left_));
- left_ = node_->left(future_right);
- return __tmp;
- }
- bool operator==(const Self &x) const {
- return left_ == x.left_;
- }
- bool operator!=(const Self &x) const {
- return left_ != x.left_;
- }
- Node *node_;
- Node *left_;
- };
- template<typename T, typename Allocator = std::allocator<T> >
- class XorList {
- private:
- typedef typename Allocator::template rebind<XorListNode<T>>::other allocator_type;
- typedef __gnu_cxx::__alloc_traits<allocator_type> Allocator_traits;
- public:
- // types:
- typedef T value_type;
- typedef typename Allocator_traits::pointer pointer;
- typedef typename Allocator_traits::const_pointer const_pointer;
- typedef value_type& reference;
- typedef const value_type& const_reference;
- typedef FwdXorListIterator<T> iterator;
- typedef std::reverse_iterator<iterator> reverse_iterator;
- typedef std::size_t size_type;
- typedef std::ptrdiff_t difference_type;
- //typedef Allocator allocator_type;
- explicit XorList(const Allocator &alloc = Allocator())
- : front_(nullptr), back_(nullptr), size_(0), allocator_(allocator_type()) {}
- XorList(size_type count, const_reference value = T(), const Allocator &alloc = Allocator())
- : front_(nullptr), back_(nullptr), size_(0), allocator_(allocator_type()) {
- for (size_type i = 0; i < count; i++)
- push_back(value);
- }
- XorList(const XorList &other) {
- *this = other;
- }
- XorList(const XorList &&other) noexcept {
- *this = other;
- }
- ~XorList() {
- XorListNode<T> *xleft = nullptr;
- XorListNode<T> *x = front_;
- while (x != back_) {
- XorListNode<T> *xnext = x->right(xleft);
- if (xnext) {
- allocator_.destroy(xleft);
- allocator_.deallocate(xleft, 1);
- }
- xleft = x;
- x = xnext;
- }
- if (xleft) {
- allocator_.destroy(xleft);
- allocator_.deallocate(xleft, 1);
- }
- if (x) {
- allocator_.destroy(x);
- allocator_.deallocate(x, 1);
- }
- }
- XorList &operator=(const XorList &other) {
- if (this == &other)
- return *this;
- this->~XorList();
- this->front_ = other.front_;
- this->back_ = other.back_;
- this->size_ = other.size_;
- this->allocator_ = other.allocator_;
- }
- XorList &operator=(const XorList &&other) noexcept {
- if (this == &other)
- return *this;
- this->~XorList();
- this->front_ = other.front_;
- this->back_ = other.back_;
- this->size_ = other.size_;
- this->allocator_ = other.allocator_;
- }
- value_type front() {
- return front_->val_;
- }
- value_type back() {
- return back_->val_;
- }
- iterator begin() {
- return iterator(front_, nullptr);
- }
- iterator end() {
- if (!back_)
- return iterator(nullptr, nullptr);
- return ++iterator(back_, back_->left(nullptr));
- }
- reverse_iterator rbegin() {
- return reverse_iterator(front_, nullptr);
- }
- reverse_iterator rend() {
- return ++reverse_iterator(back_, back_->left(nullptr));
- }
- size_type size() {
- return size_;
- }
- void push_back(const value_type &val) {
- XorListNode<T> *past_back = back_;
- back_ = reinterpret_cast<XorListNode<T> *>(allocator_.allocate(1));
- allocator_.construct(back_, std::forward<XorListNode<T> >(val));
- back_->assign_leftright(past_back, nullptr);
- if (past_back)
- past_back->assign_leftright(past_back->left(nullptr), back_);
- if (!front_)
- front_ = back_;
- size_++;
- }
- void push_back(const value_type &&val) {
- XorListNode<value_type> *past_back = back_;
- back_ = reinterpret_cast<XorListNode<T> *>(allocator_.allocate(1));
- allocator_.construct(back_, XorListNode<T>(std::forward<XorListNode<T> >(val)));
- back_->assign_leftright(past_back, nullptr);
- if (past_back)
- past_back->assign_leftright(past_back->left(nullptr), back_);
- if (!front_)
- front_ = back_;
- size_++;
- }
- void push_front(const value_type &val) {
- XorListNode<value_type> *past_front = front_;
- front_ = reinterpret_cast<XorListNode<T> *>(allocator_.allocate(1));
- allocator_.construct(front_, std::forward<XorListNode<T> >(val));
- front_->assign_leftright(nullptr, past_front);
- if (past_front)
- past_front->assign_leftright(front_, past_front->right(nullptr));
- if (!back_)
- back_ = front_;
- size_++;
- }
- void push_front(const value_type &&val) {
- XorListNode<value_type> *past_front = front_;
- front_ = reinterpret_cast<XorListNode<T> *>(allocator_.allocate(1));
- allocator_.construct(front_, std::forward<XorListNode<T> >(val));
- front_->assign_leftright(nullptr, past_front);
- if (past_front)
- past_front->assign_leftright(front_, past_front->right(nullptr));
- if (!back_)
- back_ = front_;
- size_++;
- }
- void pop_back() {
- XorListNode<T> *past_back = back_;
- back_ = back_->left(nullptr);
- back_->assign_leftright(back_->left(past_back), nullptr);
- allocator_.destroy(past_back);
- allocator_.deallocate(past_back, 1);
- }
- void pop_front() {
- XorListNode<T> *past_front = front_;
- front_ = front_->right(nullptr);
- front_->assign_leftright(nullptr, front_->right(past_front));
- allocator_.destroy(past_front);
- allocator_.deallocate(past_front, 1);
- }
- void insert_before(iterator it, T &val) {
- XorListNode<T> *cur = *it;
- XorListNode<T> *prev = it.left_;
- XorListNode<T> *past = it.node_->right(prev);
- XorListNode<T> *new_node = reinterpret_cast<XorListNode<T> *>(allocator_.allocate(1));
- allocator_.construct(new_node, std::forward<XorListNode<T> >(val));
- new_node->assign_leftright(prev, cur);
- if (cur)
- cur->assign_leftright(new_node, past);
- if (prev)
- prev->assign_leftright(prev->left(cur), new_node);
- }
- void insert_before(iterator it, T &&val) {
- XorListNode<T> *cur = *it;
- XorListNode<T> *prev = it.left_;
- XorListNode<T> *past = it.node_->right(prev);
- XorListNode<T> *new_node = allocator_.allocate(1);
- allocator_.construct(new_node, std::forward<XorListNode<T> >(val));
- new_node->assign_leftright(prev, cur);
- if (cur)
- cur->assign_leftright(new_node, past);
- if (prev)
- prev->assign_leftright(prev->left(cur), new_node);
- }
- void insert_after(iterator it, T &val) {
- XorListNode<T> *cur = *it;
- XorListNode<T> *prev = it.left_;
- XorListNode<T> *past = it.node_->right(prev);
- XorListNode<T> *new_node = allocator_.allocate(1);
- allocator_.construct(new_node, std::forward<XorListNode<T> >(val));
- new_node->assign_leftright(cur, past);
- if (cur)
- cur->assign_leftright(prev, new_node);
- if (past)
- past->assign_leftright(new_node, past->right(cur));
- }
- void insert_after(iterator it, T &&val) {
- XorListNode<T> *cur = *it;
- XorListNode<T> *prev = it.left_;
- XorListNode<T> *past = it.node_->right(prev);
- XorListNode<T> *new_node = allocator_.allocate(1);
- allocator_.construct(new_node, std::forward<XorListNode<T> >(val));
- new_node->assign_leftright(cur, past);
- if (cur)
- cur->assign_leftright(prev, new_node);
- if (past)
- past->assign_leftright(new_node, past->right(cur));
- }
- void erase(iterator it) {
- XorListNode<T> *prev = it.left_;
- XorListNode<T> *cur = it.operator->();
- XorListNode<T> *past = it.node_->right(prev);
- if (prev)
- prev->assign_leftright(prev->left(cur), past);
- if (past)
- past->assign_leftright(prev, past->right(cur));
- if (cur)
- size_--;
- allocator_.destroy(cur);
- allocator_.deallocate(cur, 1);
- }
- private:
- XorListNode<T> *front_;
- XorListNode<T> *back_; // Pointer to the LAST element
- size_type size_;
- allocator_type allocator_;
- };
- #endif //XOR_LIST_XORLIST_H
|