线段树的一种基础应用
PS:最近快期末了,在SOJ上全做的是大一都能搞定的水题,写题解的时候只能矮子里面拔将军,惭愧啊。。。
倒是为了完成集训队布置的作业,在ZOJ上发现了一两道好题,哈哈~~~
题目链接:http://cstest.scu.edu.cn/soj/problem.action?id=2163
题意:
现有一个序列是整数1~n的一种排列
对于序列中的每个数
给出这个数前面有多少个数比它小
要求输出这个序列,
算法:
用线段树上的区间[l,r]表示当前时刻序列中[l,r]区间的空位数
每个区间的初始值都等于区间的长度
倒序处理区间上的每个数
当处理到这个数的时候,
所有它右边的数均已填入空位,所有它左边的数均未填入空位
所以,找到从左边起第a[i]+1个空位所代表的数字即可。
此时这个数左边的a[i]个空位即是原序列中所有位于它左边且比它小的数
#include <stdio.h> #include <string.h> int a[40007],p[8007],ans[8007],n,i; void pushup(int rt) { a[rt]=a[rt<<1|1]+a[rt<<1]; } void build(int l,int r,int rt) { if(l==r){a[rt]=1;return;} int m; m=(l+r)>>1; build(l,m,rt<<1); build(m+1,r,rt<<1|1); pushup(rt); } void query(int l,int r,int c,int rt) { if(l==r){ans[i]=l;a[rt]=0;return;} int m; m=(l+r)>>1; if(a[rt<<1]>=c) query(l,m,c,rt<<1); else query(m+1,r,c-a[rt<<1],rt<<1|1); pushup(rt); } int main() { while(scanf("%d",&n)!=EOF) { build(1,n,1); for(i=2;i<=n;i++)scanf("%d",&p[i]); for(i=n;i>1;i--)query(1,n,p[i]+1,1); query(1,n,1,1); for(i=1;i<=n;i++)printf("%d\n",ans[i]); } }
就是明明是超级水题一道。。
做起来也就十几分钟的事儿。。。
我写起来却啰啰嗦嗦啰啰嗦嗦说不清楚