这题卡了我很久!不过也给了我很多的启示.. 因为自己对线段树的理解还是很浅显的所以这个题也便总是过不了,一直卡着卡着。遵照三鲜的原则,现在我做题极少看别人的代码,但是又没有人可以和我一起讨论的,所以今天还是看了下别人的解题报告,因为我想不出了,看着看着我发现了一个问题----->我丫的把题目读错了!太2了!区间合并线段树已经学习的差不多了,可能会在小的细节方面出点叉子,但是对基本的线段树运用的得还不错了。认真的去体会线段树的美,二叉的精巧,与构思的美妙。还是那句话:二叉树只是一个数据结构,类似于其他的数据结构一样,要有算法的辅助才能体现得出他的美。
决定要向那些大牛们学习,做一个勤奋的好孩子~虽然和他们没得什么联系... 有目标总归是好的。
在这题中学会了使用vector容器,我编码极少用STL,也没有什么机会用。不过用起来还是挺舒服的,功能强大啊~另外对于二分查找也有了新的感悟,本来在有序的序列中,就可以不用夹逼法了,定义一个维度就OK了~因为原来看错题,导致查询工作不能使用二分查找,于是便悲催得~~
另外给我的启示就是对于二叉树,不一定非得全部放在树形结构里面做,可以另外定义变量,开始我自己想思路的时候想把所有的操作都置于树形结构中做,但是发现这样做下去很困难!实现困难,查找也麻烦。后来改成了两个一维数组来额外存贮这部分的信息,于是乎,代码实现起来也简便了。所以线段树要活学活用~
加油!!
#include<stdio.h> #include<vector> using namespace std; #define MAXN 50005 struct node{ int ltemp,rtemp,mtemp; }tree[MAXN<<2]; int col[MAXN<<2]; struct block{ int s,t; block( int a,int b ){ s=a,t=b; } }; vector<block> B; int max( int a,int b ){ return a>b?a:b; } void PushUp( int rt , int m ) { tree[rt].ltemp=tree[rt<<1].ltemp; tree[rt].rtemp=tree[rt<<1|1].rtemp; if( tree[rt].ltemp==m-(m>>1) ) tree[rt].ltemp+=tree[rt<<1|1].ltemp; if( tree[rt].rtemp==m>>1 ) tree[rt].rtemp+=tree[rt<<1].rtemp; tree[rt].mtemp=max( tree[rt<<1|1].ltemp+tree[rt<<1].rtemp,max( tree[rt<<1].mtemp,tree[rt<<1|1].mtemp) ); } void PushDown( int rt,int m ) { if( col[rt]!=-1 ) { col[rt<<1]=col[rt<<1|1]=col[rt]; tree[rt<<1].ltemp=tree[rt<<1].mtemp=tree[rt<<1].rtemp=col[rt]?0:(m-(m>>1)); tree[rt<<1|1].ltemp=tree[rt<<1|1].mtemp=tree[rt<<1|1].rtemp=col[rt]?0:(m>>1); col[rt]=-1; } } void build( int l,int r,int rt ) { col[rt]=-1; tree[rt].ltemp=tree[rt].mtemp=tree[rt].rtemp=r-l+1; if( l==r )return ; int m=( l+r )>>1; build( l,m,rt<<1 ); build( m+1,r,rt<<1|1 ); } int query( int w,int l,int r,int rt ) { int m=( l+r )>>1; if( l==r ) return l; PushDown( rt,r-l+1 ); if( tree[rt<<1].mtemp>=w ) return query( w,l,m,rt<<1 ); else if( tree[rt<<1|1].ltemp+tree[rt<<1].rtemp>=w ) return m-tree[rt<<1].rtemp+1; else return query( w,m+1,r,rt<<1|1 ); } void update( int l,int r,int c,int L,int R,int rt ) { if( l<=L&&R<=r ) { col[rt]=c; tree[rt].ltemp=tree[rt].mtemp=tree[rt].rtemp=col[rt]?0:R-L+1; return ; } PushDown( rt,R-L+1 ); int m=( L+R )>>1; if( l<=m ) update( l,r,c,L,m,rt<<1 ); if( r>m ) update( l,r,c,m+1,R,rt<<1|1 ); PushUp( rt,R-L+1 ); } int Bin( int at ) { int left=0;int right=B.size()-1; int m; while( left<=right ) { m=( left+right )>>1; if( at>=B[m].s ) left=m+1; else right=m-1; } return left; } int main() { int n,m; char com[10]; int date,i; while( scanf("%d %d",&n,&m)!=EOF ) { build( 1,n,1 ); B.clear(); while( m-- ) { scanf( "%s",&com ); if( com[0]=='N' ) { scanf( "%d",&date ); if( date>tree[1].mtemp ) printf( "Reject New\n" ); else { int at=query( date,1,n,1 ); block k(at,at+date-1); int index=Bin(at);//printf( "%d\n",index ); B.insert( B.begin()+index,k ); update( at,at+date-1,1,1,n,1 ); printf( "New at %d\n",at ); } } else if( com[0]=='R' ) { update(1,n,0,1,n,1); B.clear(); printf( "Reset Now\n" ); } else if( com[0]=='G' ) { scanf( "%d",&date ); if( date>B.size() ) { printf( "Reject Get\n" ); continue; } else printf( "Get at %d\n",B[date-1].s ); } else if( com[0]=='F' ) { scanf( "%d",&date ); int index=Bin(date)-1; //printf( "%d\n",index ); if( index!=-1 && B[index].t>=date ) { printf( "Free from %d to %d\n",B[index].s,B[index].t ); update( B[index].s,B[index].t,0,1,n,1 ); B.erase( B.begin()+index ); } else printf( "Reject Free\n" ); } } printf( "\n" ); } return 0; }