zoukankan      html  css  js  c++  java
  • HDU 2852 KiKi's K-Number 主席树

    题意:

    要求维护一个数据结构,支持下面三种操作:

    • (0 \, e):插入一个值为(e)的元素
    • (1 \, e):删除一个值为(e)的元素
    • (2 \, a \, k):查询比(a)大的数中的第(k)

    分析:

    因为插入的数在(10^5)内,所以不需要离散化。
    维护一棵主席树,每次插入或者删除元素都新建一棵主席树。
    对于查询操作,先查询不超过(a)的元素的个数(cnt),然后再查询序列中第(cnt+k)小即可。

    #include <cstdio>
    #include <cstring>
    #include <algorithm>
    using namespace std;
    
    const int maxn = 100000 + 10;
    const int maxnode = (maxn << 5);
    
    int n;
    
    int sz, root[maxn];
    int lch[maxnode], rch[maxnode], sum[maxnode];
    
    int update(int pre, int L, int R, int p, int v) {
    	int rt = ++sz;
    	sum[rt] = sum[pre] + v;
    	if(L < R) {
    		int M = (L + R) / 2;
    		if(p <= M) { rch[rt] = rch[pre]; lch[rt] = update(lch[pre], L, M, p, v); }
    		else { lch[rt] = lch[pre]; rch[rt] = update(rch[pre], M+1, R, p, v); }
    	}
    	return rt;
    }
    
    int Qcnt(int rt, int L, int R, int p) {
    	if(L == R) return sum[rt];
    	int M = (L + R) / 2;
    	if(p <= M) return Qcnt(lch[rt], L, M, p);
    	else return Qcnt(rch[rt], M+1, R, p);
    }
    
    int Qsum(int rt, int L, int R, int p) {
    	if(L == R) return sum[rt];
    	int M = (L + R) / 2;
    	if(p <= M) return Qsum(lch[rt], L, M, p);
    	else return sum[lch[rt]] + Qsum(rch[rt], M+1, R, p);
    }
    
    int kth(int rt, int L, int R, int k) {
    	if(L == R) return L;
    	int num = sum[lch[rt]];
    	int M = (L + R) / 2;
    	if(num >= k) return kth(lch[rt], L, M, k);
    	else return kth(rch[rt], M+1, R, k-num);
    }
    
    int main()
    {
    	while(scanf("%d", &n) == 1) {
    		int cnt = 0;
    		sz = 0;
    		while(n--) {
    			int op, a; scanf("%d%d", &op, &a);
    			if(op == 0) {
    				cnt++;
    				root[cnt] = update(root[cnt-1], 1, maxn, a, 1);
    			} else if(op == 1) {
    				if(!Qcnt(root[cnt], 1, maxn, a)) puts("No Elment!");
    				else {
    					cnt++;
    					root[cnt] = update(root[cnt-1], 1, maxn, a, -1);
    				}
    			} else {
    				int b; scanf("%d", &b);
    				int k = Qsum(root[cnt], 1, maxn, a);
    				//printf("k1 = %d
    ", k);
    				k += b;
    				//printf("k2 = %d
    ", k);
    				if(k > sum[root[cnt]]) { puts("Not Find!"); continue; }
    				printf("%d
    ", kth(root[cnt], 1, maxn, k));
    			}
    		}
    	}
    
    	return 0;
    }
    
  • 相关阅读:
    pycharm中文乱码
    bootstrap-table分页
    sql将查询结果的某个字段赋值给另一个字段
    bootstrap tab选项卡
    cocos-js 精灵移动转圈
    配置环境变量路径有空格
    配置java环境遇到的问题及解决方案
    obj = obj || {} 分析这个代码的起到的作用
    sql server行转列
    sql server统计总成绩和排名
  • 原文地址:https://www.cnblogs.com/AOQNRMGYXLMV/p/5343355.html
Copyright © 2011-2022 走看看