diff options
| author | mzuenni <michi.zuendorf@gmail.com> | 2022-06-27 17:19:28 +0200 |
|---|---|---|
| committer | mzuenni <michi.zuendorf@gmail.com> | 2022-06-27 17:19:28 +0200 |
| commit | 5ab8a5088b729a9953b8dff1b2a985dc8fb2098b (patch) | |
| tree | ed40d6936c0e9eee40ba62751cbf99ecddbaddc2 /datastructures/treap.cpp | |
| parent | adabbad9c51cf7cd3874bfde8eac1fbcf84fec10 (diff) | |
updated tcr
Diffstat (limited to 'datastructures/treap.cpp')
| -rw-r--r-- | datastructures/treap.cpp | 98 |
1 files changed, 69 insertions, 29 deletions
diff --git a/datastructures/treap.cpp b/datastructures/treap.cpp index 4de3328..6296392 100644 --- a/datastructures/treap.cpp +++ b/datastructures/treap.cpp @@ -1,39 +1,79 @@ -struct item { - int key, prior; - item *l, *r; - item() {} - item(int key, int prior) : key(key), prior(prior), l(NULL), r(NULL) {} +struct node { + int key, prio, left, right, size; + node(int key, int prio) : key(key), prio(prio), left(-1), + right(-1), size(1) {}; }; -void split(item *t, int key, item *l, item *r) { - if (!t) l = r = NULL; - else if (key < t->key) split(t->l, key, l, t->l), r = t; - else split(t->r, key, t->r, r), l = t; -} +vector<node> treap; -void insert(item *t, item *it) { - if (!t) t = it; - else if (it->prior > t->prior) split(t, it->key, it->l, it->r), t = it; - else insert(it->key < t->key ? t->l : t->r, it); +int getSize(int root) { + return root < 0 ? 0 : treap[root].size; } -void merge(item *t, item *l, item *r) { - if (!l || !r) t = l ? l : r; - else if (l->prior > r->prior) merge(l->r, l->r, r), t = l; - else merge(r->l, l, r->l), t = r; +void update(int root) { + if (root < 0) return; + treap[root].size = 1 + getSize(treap[root].left) + + getSize(treap[root].right); } -void erase(item *t, int key) { - if (t->key == key) merge (t, t->l, t->r); - else erase(key < t->key ? t->l : t->r, key); +pair<int, int> split(int root, int minKeyRight) { + if (root < 0) return {-1, -1}; + if (treap[root].key >= minKeyRight) { + auto leftSplit = split(treap[root].left, minKeyRight); + treap[root].left = leftSplit.second; + update(root); + leftSplit.second = root; + return leftSplit; + } else { + auto rightSplit = split(treap[root].right, minKeyRight); + treap[root].right = rightSplit.first; + update(root); + rightSplit.first = root; + return rightSplit; +}} + +int merge (int left, int right) { + if (left < 0) return right; + if (right < 0) return left; + if (treap[left].prio < treap[right].prio) { //min priority heap + treap[left].right = merge(treap[left].right, right); + update(left); + return left; + } else { + treap[right].left = merge(left, treap[right].left); + update(right); + return right; +}} + +//insert values with high priority first +int insert(int root, int key, int prio) { + int next = sz(treap); + treap.emplace_back(key, prio); + auto t = split(root, key); + //returns new root + return merge(merge(t.first, next), t.second); } -item *unite(item *l, item *r) { - if (!l || !r) return l ? l : r; - if (l->prior < r->prior) swap(l, r); - item * lt, rt; - split(r, l->key, lt, rt); - l->l = unite(l->l, lt); - l->r = unite(l->r, rt); - return l; +int remove(int root, int key) { + if (root < 0) return -1; + if (key < treap[root].key) { + treap[root].left = remove(treap[root].left, key); + update(root); + return root; + } else if (key > treap[root].key) { + treap[root].right = remove(treap[root].right, key); + update(root); + return root; + } else { //check prio? + return merge(treap[root].left, treap[root].right); +}} + +int kth(int root, int k) { + if (root < 0) return -1; + int leftSize = getSize(treap[root].left); + if (k < leftSize) return kth(treap[root].left, k); + else if (k > leftSize) { + return kth(treap[root].right, k - 1 - leftSize); + } + return root; } |
