codevs 1191 线段树 区间更新(水)

时间:2023-03-09 22:52:43
codevs 1191 线段树 区间更新(水)

题目描述 Description

在一条数轴上有N个点,分别是1~N。一开始所有的点都被染成黑色。接着
我们进行M次操作,第i次操作将[Li,Ri]这些点染成白色。请输出每个操作执行后
剩余黑色点的个数。

输入描述 Input Description

输入一行为N和M。下面M行每行两个数Li、Ri

输出描述 Output Description

输出M行,为每次操作后剩余黑色点的个数。

样例输入 Sample Input

10 3
3 3
5 7
2 8

样例输出 Sample Output

9
6
3

数据范围及提示 Data Size & Hint

数据限制
对30%的数据有1<=N<=2000,1<=M<=2000
对100%数据有1<=Li<=Ri<=N<=200000,1<=M<=200000

题意:中文题面

题解:区间更新 输出tree[1].sum;

if(tree[root].sum==0) return;//这行代码是关键

 #include<iostream>
#include<cstring>
#include<cstdio>
#include<queue>
#include<stack>
#define ll long long
#define maxn 200005
using namespace std;
struct node
{
int l,r;
int sum;
}tree[*maxn];
int x,y;
int n,q;
int ans=;
void buildtree(int root,int left,int right)
{
tree[root].l=left;
tree[root].r=right;
if(left==right)
{
tree[root].sum=;
return ;
}
int mid=(left+right)>>;
buildtree(root<<,left,mid);
buildtree(root<<|,mid+,right);
tree[root].sum=tree[root<<].sum+tree[root<<|].sum;
}
void updata(int root,int left,int right,int c)
{
if(tree[root].sum==) return;
if(tree[root].l==left&&tree[root].r==right)
{
tree[root].sum=;
return ;
}
if(tree[root].l==tree[root].r)
return ;
int mid=(tree[root].l+tree[root].r)>>;
if(right<=mid)
updata(root<<,left,right,c);
else
{
if(left>mid)
updata(root<<|,left,right,c);
else
{
updata(root<<,left,mid,c);
updata(root<<|,mid+,right,c);
}
}
tree[root].sum=tree[root<<].sum+tree[root<<|].sum;
}
int main()
{
scanf("%d %d",&n,&q);
buildtree(,,n);
for(int j=;j<=q;j++)
{
scanf("%d %d",&x,&y);
updata(,x,y,);
printf("%d\n",tree[].sum);} return ;
}