#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define ll long long
#define ull unsigned long long
#define cls(x) memset(x,0,sizeof(x))
#define clslow(x) memset(x,-1,sizeof(x))
#define INF 0x3f3f3f3f
using namespace std;
const int maxn=2e4+100;
const int mod=1e9+7;
int a[maxn<<1],lazy[maxn<<3],tot=1,vis[maxn],Count=0;
struct nn
{
int l,r;
}p[maxn];
void Push_down(int rt)
{
if(lazy[rt]!=-1){
lazy[rt<<1]=lazy[rt];
lazy[rt<<1|1]=lazy[rt];
lazy[rt]=-1;
}
}
void Update(int L,int R,int l,int r,int rt,int C)
{
if(L<=l && r<=R)
{
lazy[rt]=C;
return;
}
int mid=(r+l)>>1;
Push_down(rt);
if(L<=mid)Update(L,R,l,mid,rt<<1,C);
if(R>mid)Update(L,R,mid+1,r,rt<<1|1,C);
}
void Query(int L,int R,int l,int r,int rt)
{
if(L<=l && r<=R && lazy[rt] != -1 )
{
if(vis[ lazy[rt] ] == 0)
vis[ lazy[rt] ] = 1,
Count++;
return ;
}
if(l==r)return;
int mid=(l+r)>>1;
if(L<=mid)Query(L,R,l,mid,rt<<1);
if(R>mid)Query(L,R,mid+1,r,rt<<1|1);
}
int main()
{
int t;
scanf("%d",&t);
while(t--){
memset(vis,0,sizeof vis);
memset(lazy,-1,sizeof lazy);
Count=0;
tot=1;
int n;
scanf("%d",&n);
for(int i=1;i<=n;i++){
scanf("%d%d",&p[i].l,&p[i].r);
a[tot++]=p[i].l;
a[tot++]=p[i].r;
}
sort(a+1,a+tot);
tot=unique(a+1,a+tot)-a;
int k=tot;
for(int i=2;i<k;i++){
if(a[i]-a[i-1]>1)a[tot++]=a[i-1]+1;
}
sort(a+1,a+tot);
for(int i=1;i<=n;i++){
int l,r;
l=lower_bound(a+1,a+tot,p[i].l)-a;
r=lower_bound(a+1,a+tot,p[i].r)-a;
Update(l,r,1,tot-1,1,i);
}
离散化上个博客说了,这题的离散化不能全部沿用上一个博客的,有特例当一二三副海报分别是 1----10 ,1----4 ,6----10 的时候,离散化完就是a[1]=1,a[2]=4,a[3]=6,a[4]=10,第一张海报贴上后线段树更新1到4全部为1,第二张海报贴上后1到2都为2,第三上贴上后3到4都为3,最后统计查询的时候,输出结果就是2,第一张海报莫名被遮住了,所以需要另一种离散化,当两个数字只差大于1,加一个数字进入二者中间,如上例子,加完数字就变成a[1]=1 ,a[2]=2 , a[3]=4 , a[4]=5 , a[5]=6 , a[6]=7 , a[7]=10 黄色的就是加上去防止覆盖的数字。
线段树要用的只有更新和查询操作,直接用懒惰数组就可以完成了,还有细节在代码注释里。