POJ线段树第一题,这一上来就是成段更新加离散化,这训练计划要闹哪样~~~~~
题目大意:
一面墙上贴海报,问最后能看到多少张海报。因为墙的宽度是单位长度,海报的宽度也是单位长度,所以我们只要来看墙和海报的长度就行。
解题思路:
利用线段树可以解决。
注意因为海报的区间太大,我们得对所有端点离散化,注意如果端点排序之后相邻两点不连续需要在中间再加一个点。
下面是代码:
#include <stdio.h> #include <string.h>; #include <algorithm> using namespace std; const int Max =200005; int node[Max<<2]; struct node1 { int l,r; } paper[Max]; int num[Max<<2],cnt; bool vis[Max<<2]; int Bsearch(int key) { int high=cnt-1; int low=0,mid; while(low<=high) { mid=(low+high)>>1; if(num[mid]<key)low=mid+1; else if(num[mid]==key)return mid; else high=mid-1; } return 0; } void Pushdown(int tr) { if(node[tr]) { node[tr<<1]=node[tr]; node[tr<<1|1]=node[tr]; node[tr]=0; } } void update(int L,int R,int col,int l,int r,int tr) { if(L<=l&&r<=R) { node[tr]=col; return; } Pushdown(tr); int m=(l+r)>>1; if(L<=m)update(L,R,col,l,m,tr<<1); if(R>m)update(L,R,col,m+1,r,tr<<1|1); } int query(int l,int r,int tr) { int nct=0; if(node[tr]) { if(!vis[node[tr]]) { nct=1; } vis[node[tr]]=true; return nct; } if(l==r)return 0; int m=(l+r)>>1; int ans=query(l,m,tr<<1); ans+=query(m+1,r,tr<<1|1); return ans; } int main() { int t; scanf("%d",&t); while(t--) { int n,m=1,l,r; cnt=0; memset(vis,0,sizeof(vis)); memset(node,0,sizeof(node)); scanf("%d",&n); for(int i=0; i<n; i++) { scanf("%d%d",&paper[i].l,&paper[i].r); num[cnt++]=paper[i].l; num[cnt++]=paper[i].r; } sort(num,num+cnt); for(int i=1; i<cnt; i++) { if(num[i]!=num[i-1])num[m++]=num[i]; } for(int i=m-1; i>0; i--) { if(num[i]!=num[i-1]+1) num[m++]=num[i-1]+1; } sort(num,num+m); cnt=m; for(int i=0; i<n; i++) { l=Bsearch(paper[i].l)+1; r=Bsearch(paper[i].r)+1; update(l,r,i+1, 1,m,1); } printf("%d\n",query(1,m,1)); } return 0; }