日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程资源 > 编程问答 >内容正文

编程问答

P2596 [ZJOI2006]书架(fhq treap)

發(fā)布時(shí)間:2023/12/4 编程问答 31 豆豆
生活随笔 收集整理的這篇文章主要介紹了 P2596 [ZJOI2006]书架(fhq treap) 小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

P2596 [ZJOI2006]書架

我們用fhq treap來(lái)完成這一題

對(duì)于一個(gè)新插入的節(jié)點(diǎn)我們?nèi)?quán)值為其索引值,其所記錄的valuevaluevalue是其當(dāng)前索引所在位置。

操作一:把索引為valuevaluevalue的點(diǎn)放到平衡樹(shù)前面,分別別得到三顆子樹(shù)x,y,zx, y, zx,y,z(前段子樹(shù),索引為valuevaluevalue所代表的子樹(shù),后段子樹(shù)),同時(shí)把val[y]val[y]val[y]修改成全局最小,然后按照順序merge(y,x,z)merge(y, x, z)merge(y,x,z)

操作二:與操作一類似得到x,y,zx, y, zx,y,z,然后把val[y]val[y]val[y]改成全局最大,按照順序merge(x,z,y)merge(x, z, y)merge(x,z,y)

操作三:t=0t = 0t=0不做操作,t=1t = 1t=1得到四顆子樹(shù)x,y,z,wx, y, z, wx,y,z,w(前段子樹(shù),valuevaluevalue所代表的子樹(shù),valuevaluevalue的后繼,后端子樹(shù)),交換信息,然后按照順序merge(x,z,y,w)merge(x, z, y, w)merge(x,z,y,w)t=?1t = -1t=?1得到四顆子樹(shù)x,y,z,wx, y, z, wx,y,z,w(前段子樹(shù),valuevaluevalue的前驅(qū),valuevaluevalue所代表的子樹(shù),后端子樹(shù)),交換信息,然后按照順序merge(x,z,y,w)merge(x, z, y, w)merge(x,z,y,w)

操作四:按照值分裂成兩棵樹(shù),然后輸出第一顆樹(shù)的大小即可。

操作五:直接查找第kkk大即可。

#include <bits/stdc++.h>using namespace std;const int N = 1e5 + 10;mt19937 rnd(233);int minn, maxn;struct Tree {int ls[N], rs[N], val[N], sz[N], key[N], root, cnt;void push_up(int rt) {sz[rt] = sz[ls[rt]] + sz[rs[rt]] + 1;}int new_node(int value, int pos) {val[value] = pos, sz[value] = 1, key[value] = rnd();return value;}void split(int rt, int value, int &x, int &y) {if (!rt) {x = y = 0;return ;}if (val[rt] <= value) {x = rt;split(rs[rt], value, rs[rt], y);}else {y = rt;split(ls[rt], value, x, ls[rt]);}push_up(rt);}int merge(int x, int y) {if (!x || !y) {return x | y;}if (key[x] < key[y]) {rs[x] = merge(rs[x], y);push_up(x);return x;}else {ls[y] = merge(x, ls[y]);push_up(y);return y;}}int get_num(int rt, int rank) {while (rt) {if (sz[ls[rt]] + 1 == rank) {break;}if (sz[ls[rt]] >= rank) {rt = ls[rt];}else {rank -= sz[ls[rt]] + 1;rt = rs[rt];}}return rt;}int get_num(int rank) {return get_num(root, rank);}void insert(int value, int pos) {int x, y;split(root, pos, x, y);root = merge(merge(x, new_node(value, pos)), y);}void update(int value, int op) {int x, y, z;split(root, val[value], x, z);split(x, val[value] - 1, x, y);if (op) {val[value] = --minn;root = merge(merge(y, x), z);}else {val[value] = ++maxn;root = merge(merge(x, z), y);}}void reverse(int value, int op) {if (!op) {return ;}if (op == 1) {int x, y, z, w;split(root, val[value], x, z);split(x, val[value] - 1, x, y);int t = get_num(z, 1);split(z, val[t], z, w);swap(val[y], val[z]);root = merge(merge(x, z), merge(y, w));}else {int x, y, z, w;split(root, val[value] - 1, x, z);split(z, val[value], z, w);int t = get_num(x, sz[x]);split(x, val[t] - 1, x, y);swap(val[y], val[z]);root = merge(merge(x, z), merge(y, w));}}int get_rank(int value) {int x, y, ans;split(root, val[value] - 1, x, y);ans = sz[x];root = merge(x, y);return ans;} }tree;int main() {// freopen("in.txt", "r", stdin);// freopen("out.txt", "w", stdout);// ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);int n, m;scanf("%d %d", &n, &m);minn = 1, maxn = n;for (int i = 1, x; i <= n; i++) {scanf("%d", &x);tree.insert(x, i);}char op[10];for (int i = 1, s, t; i <= m; i++) {scanf("%s %d", op, &s);if (op[0] == 'T') {tree.update(s, 1);}else if (op[0] == 'B') {tree.update(s, 0);}else if (op[0] == 'I') {scanf("%d", &t);tree.reverse(s, t);}else if (op[0] == 'A') {printf("%d\n", tree.get_rank(s));}else {printf("%d\n", tree.get_num(s));}}return 0; }

總結(jié)

以上是生活随笔為你收集整理的P2596 [ZJOI2006]书架(fhq treap)的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。