标签:
Time Limit: 20000MS | Memory Limit: 65536K | |
Total Submissions: 50247 | Accepted: 17101 | |
Case Time Limit: 2000MS |
Description
Input
Output
Sample Input
7 3 1 5 2 6 3 7 4 2 5 3 4 4 1 1 7 3
Sample Output
5 6 3
Hint
题目链接:POJ 2104 HDU 2665(HDU题面描述有误,虽说是求kth big其实还是跟POJ一样求kth small)
看主席树看了N久,终于把更新操作和查询操作看懂了:),其间找了无数篇博客但是其中绝大部分都是泛泛而谈有一些甚至只有代码连注释都没有,
很多次想弃疗(感觉这高大上的数据结构弱比我估计以后很难用到),然而还是没有放弃终于看懂一点点……故想通过这两道模版题来说一下弱比我的理解……
首先要知道普通线段树如何求整个范围内第k大,单点更新+单点查询(至叶子节点),然后就是这个区间第K大(小)怎么弄呢?由于单点更新的时候只会更新一条路上的东西即树上一层只会更新一个节点,比如某一层有两个节点来管理【1,2】与【 3,4】,假设更新1这个值,显然【3,4】是不会变的,只是管理【1,2】这个节点的cnt值加1,当然,这个节点下面还有【1,1】与【2,2】这俩叶子,同样道理也只会更新管理【1,1】的节点,使它的cnt值加1,管理【2,2】的节点不动……
根据二叉树的性质可以知道,每一次修改就只要新建出Log(N)左右个节点就可以与之前的树形成一个新的树,树上的老节点就是不用更新的,新节点就是更新的再指向老的节点(即很可能会与老节点共用子节点)。然后就用到了前缀和的思想,一个元素均不相同的序列可以发现这样一个规律,假设10个数为1 3 5 7 9 2 4 6 8 10,先简化一下问题,求【3,7】的节点cnt为多少,那显然是【1,7】.cnt-【1,(3-1)】.cnt=4,继而就可以求【3,4】、【5,7】甚至更后面的小区间内cnt值,那问题就成了一个前面讲的线段树求整个范围内第k大,只是整个范围一直被维护为L,R之间而已。
主席树这种写法需要root数组、和lson与rson的数组,还有记录区间内数字出现个数的cnt数组
更新操作:首先复制当前所在原节点的所有信息,再将cnt+1(因为你在原来的基础上插入了一个数),然后用单点更新的思路进行复制并更新(讲复制感觉更贴切实际,因为更新是在副本上更新的而不是原来节点上),走到哪就把哪个节点复制下来再在这个副本上进行更新
查询操作:用两颗不同的树的cnt作差再判断查询方向,写法上跟普通的单点查询非常相似
最后题目数据范围比较大需要离散化一下另外我加了个没什么卵用的读入外挂……
POJ 2104代码:
#include <stdio.h> #include <iostream> #include <algorithm> #include <cstdlib> #include <cstring> #include <vector> using namespace std; #define INF 0x3f3f3f3f #define CLR(x,y) memset(x,y,sizeof(x)) #define LC(x) (x<<1) #define RC(x) ((x<<1)+1) #define MID(x,y) ((x+y)>>1) typedef pair<int,int> pii; typedef long long LL; const int N=100010; struct seg { int lson,rson; int cnt; }; seg T[N*20]; int tot,rt[N]; int arr[N]; vector<int>pos; void init() { tot=0; CLR(rt,0); pos.clear(); } int Scan() { int res=0,ch,flag=0; if((ch=getchar())==‘-‘) flag=1; else if(ch>=‘0‘&&ch<=‘9‘) res=ch-‘0‘; while((ch=getchar())>=‘0‘&&ch<=‘9‘) res=res*10+ch-‘0‘; return flag?-res:res; } void build(int &cur,int l,int r)//注意第一个参数是引用为了修改原值 { cur=++tot; T[cur].cnt=0; if(l==r) return ; else { int mid=MID(l,r); build(T[cur].lson,l,mid); build(T[cur].rson,mid+1,r); } } void update(int &cur,int ori,int l,int r,int val)//cur为准备新开的节点,ori为原来对应位置的节点 { cur=++tot;//新建一颗树 T[cur].lson=T[ori].lson;//复制信息 T[cur].rson=T[ori].rson;//复制信息 T[cur].cnt=T[ori].cnt+1;//更新新增值 if(l==r) return ; else { int mid=MID(l,r); if(val<=mid) update(T[cur].lson,T[ori].lson,l,mid,val); else update(T[cur].rson,T[ori].rson,mid+1,r,val); } } int query(int st,int ed,int l,int r,int k) { if(l==r) return l; else { int mid=MID(l,r); int cnt=T[T[ed].lson ].cnt-T[T[st].lson ].cnt;//与普通求Kth一样先看lson(看rson也行下面顺序变一下而已) if(k<=cnt) return query(T[st].lson,T[ed].lson,l,mid,k); else return query(T[st].rson,T[ed].rson,mid+1,r,k-cnt); } } int main(void) { int n,m,i,l,r,k; while (~scanf("%d%d",&n,&m)) { for (i=1; i<=n; ++i) { scanf("%d",&arr[i]); pos.push_back(arr[i]); } sort(pos.begin(),pos.end());//pos储存着原值 pos.erase(unique(pos.begin(),pos.end()),pos.end()); for (i=1; i<=n; ++i) arr[i]=lower_bound(pos.begin(),pos.end(),arr[i])-pos.begin()+1;//arr变为离散化之后的数组仅用来更新 int SZ=pos.size(); build(rt[0],1,SZ); for (i=1; i<=n; ++i) update(rt[i],rt[i-1],1,SZ,arr[i]); for (i=0; i<m; ++i) { l=Scan(); r=Scan(); k=Scan(); int indx=query(rt[l-1],rt[r],1,SZ,k)-1;//由于pos下标从0开始 printf("%d\n",pos[indx]); } init(); } return 0; }
POJ 2104&HDU 2665 Kth number(主席树入门+离散化)
标签:
原文地址:http://www.cnblogs.com/Blackops/p/5929126.html