首页 > 其他分享 >[NOIP2022] 比赛

[NOIP2022] 比赛

时间:2022-12-12 22:00:31浏览次数:57  
标签:lazy 比赛 int Tree long NOIP2022 clox cloy

[NOIP2022] 比赛

很明显要离线,枚举右端点

对于\([L,R]\)中取子区间的最大值求和不是很好维护

定义\(f_l\)为当前左端点为\(l\),\(\sum\limits_{r=l}^RAns(l,r)\)

\(Query(l,r)\)就是\(\sum\limits_{i=l}^rf_i\),我们可以用线段树维护

定义\(ma_l\)为当前\(R\),\(A\)中\((l,R)\)的最大值,\(mb_l\)同理

考虑我们加入一个\(R\)对\(f,ma,mb\)的影响

很明显,\(ma,mb\)就是与\(A_R,B_R\)取\(max\),又因为\(ma,mb\)单增为了方便我们找到第一个\(R\)大于的用区间覆盖维护

对于\(f\),其实就相当于对于\([1,R]\)所有的\(f_l+=(ma_l\times mb_l)\)

令\(ma_l=x,mb_l=y\)

如果我们把线段树节点上的\(f\)拆分为\(c_{xy}\sum xy+c_x\sum x+c_y\sum y+c\sum1\),则上面的操作到线段树上就是\(c_{xy}+=1\)

而如果是区间覆盖实际上是可以做这样的拆分的:

\[Case1:A,B都没被覆盖 \\ 没影响 \\ Case2:A覆盖,B没 \\ 则xy,x要变,其中的c_{xy}\sum{xy}=c_{xy}x'\sum y,相当于这一项的系数到y上 \\ c_x到常数项 \\ Case3:B覆盖,A没 \\ 同理 \\ Case4:A,B都被覆盖 \\ 都到常数项 \]

由此,覆盖操作经拆分依旧可以互相转化

最后就是打标记的问题了,因为上面的我们要系数的增量,如果当前儿子有覆盖标记的话,下传的增量标记是要转化的

这里我们规定先传增量再传覆盖,这样保证所有增量在覆盖前面,维护的\(\sum xy\)对应的增量也是未覆盖的

#include<bits/stdc++.h>
#define int unsigned long long
#define ls 2*p
#define rs 2*p+1
using namespace std;
int T,n;
const int MAXN=2.5e5+5;
int a[MAXN];
int b[MAXN];
int q;
unsigned long long Ans[MAXN];
vector<pair<int,int> >query[MAXN];
int l,r;
struct Tag{
	unsigned long long addx,addy,addxy,addc;
	int clox,cloy;
};
struct Seg{
	unsigned long long datex,datey,datexy,date;
	int l,r;
	Tag lazy;
}Tree[MAXN*4];
void Build(int p,int l,int r)
{
	Tree[p].l=l;
	Tree[p].r=r;
	if(l==r)
	{
		return;
	}
	int mid=(l+r)>>1;
	Build(ls,l,mid);
	Build(rs,mid+1,r);
}
void Put(Seg &u,int Len,Tag t)
{
	if(u.lazy.clox&&u.lazy.cloy)
	{
		u.lazy.addc+=t.addxy*u.lazy.clox*u.lazy.cloy+t.addx*u.lazy.clox+t.addy*u.lazy.cloy+t.addc;
	}
	else if(u.lazy.clox)
	{
		u.lazy.addy+=t.addxy*u.lazy.clox+t.addy;
		u.lazy.addc+=t.addx*u.lazy.clox+t.addc;
	}
	else if(u.lazy.cloy)
	{
		u.lazy.addx+=t.addxy*u.lazy.cloy+t.addx;
		u.lazy.addc+=t.addy*u.lazy.cloy+t.addc;
	}
	else
	{
		u.lazy.addxy+=t.addxy;
		u.lazy.addx+=t.addx;
		u.lazy.addy+=t.addy;
		u.lazy.addc+=t.addc;
	}
	if(t.clox)
	{
		u.lazy.clox=t.clox;
	}
	if(t.cloy)
	{
		u.lazy.cloy=t.cloy;
	}
	
	u.date+=u.datex*t.addx+u.datey*t.addy+u.datexy*t.addxy+Len*t.addc;
	if(t.clox&&t.cloy)
	{
		u.datex=t.clox*Len;
		u.datey=t.cloy*Len;
		u.datexy=t.clox*t.cloy*Len;
	}
	else if(t.clox)
	{
		u.datex=t.clox*Len;
		u.datexy=u.datey*t.clox;
	}
	else if(t.cloy)
	{
		u.datey=t.cloy*Len;
		u.datexy=u.datex*t.cloy;
	}
}
void push_up(int p)
{
	Tree[p].date=Tree[ls].date+Tree[rs].date;
	Tree[p].datex=Tree[ls].datex+Tree[rs].datex;
	Tree[p].datey=Tree[ls].datey+Tree[rs].datey;
	Tree[p].datexy=Tree[ls].datexy+Tree[rs].datexy;
}
void Clear(Tag &x)
{
	x.addc=0;
	x.addx=0;
	x.addy=0;
	x.addxy=0;
	x.clox=0;
	x.cloy=0;
	return;
}
Tag Make(int clox,int cloy,unsigned long long addx,unsigned long long addy,unsigned long long addc,unsigned long long addxy)
{
	Tag nyh520;
	nyh520.clox=clox;
	nyh520.cloy=cloy;
	nyh520.addx=addx;
	nyh520.addy=addy;
	nyh520.addc=addc;
	nyh520.addxy=addxy;
	return nyh520;
}
void push_down(int p)
{
	Put(Tree[ls],Tree[ls].r-Tree[ls].l+1,Tree[p].lazy);
	Put(Tree[rs],Tree[rs].r-Tree[rs].l+1,Tree[p].lazy);
	Clear(Tree[p].lazy);
//	if((Tree[p].lazy.clox||Tree[p].lazy.cloy))
//	{
//		printf("why\n");
//	}
}
void Update(int p,int l,int r,Tag x)
{
	if(Tree[p].l>=l&&Tree[p].r<=r)
	{
		Put(Tree[p],Tree[p].r-Tree[p].l+1,x);
		return;
	}
	push_down(p);
	int mid=(Tree[p].l+Tree[p].r)>>1;
	if(l<=mid)
	{
		Update(ls,l,r,x);
	}
	if(r>mid)
	{
		Update(rs,l,r,x);
	}
	push_up(p);
}
int twice;
unsigned long long Query(int p,int l,int r)
{
	if(Tree[p].l>=l&&Tree[p].r<=r)
	{
		return Tree[p].date;
	}
//	if(twice&&(Tree[p].lazy.clox||Tree[p].lazy.cloy))
//	{
//		printf("wdg\n");
//	}
	push_down(p);
//	if((Tree[p].lazy.clox||Tree[p].lazy.cloy))
//	{
//		printf("why\n");
//	}
	int mid=(Tree[p].l+Tree[p].r)>>1;
	unsigned long long Res=0;
	if(l<=mid)
	{
		Res+=Query(ls,l,r);
	 } 
	if(r>mid)
	{
		Res+=Query(rs,l,r);
	}
	push_up(p);
	return Res;
}
int dp1[MAXN][25];
int dp2[MAXN][25];
int Query1(int x,int y)
{
    int k=log2(y-x+1);
    return max(dp1[x][k],dp1[y-(1<<k)+1][k]);
}
int Query2(int x,int y)
{
    int k=log2(y-x+1);
    return max(dp2[x][k],dp2[y-(1<<k)+1][k]);
}
signed main()
{
	scanf("%llu %llu",&T,&n);
	for(int i=1;i<=n;i++)
	{
		scanf("%llu",&a[i]);	
	}	
	for(int i=1;i<=n;i++)
	{
		scanf("%llu",&b[i]);
	}
	for(int i=1;i<=n;i++)
	{
		dp1[i][0]=a[i];
	}
	for(int j=1;(1<<j)<=n;j++)
	{
		for(int i=1;i+(1<<j)-1<=n;i++)
		{
			dp1[i][j]=max(dp1[i][j-1],dp1[i+(1<<j-1)][j-1]);
		}
	}
	for(int i=1;i<=n;i++)
	{
		dp2[i][0]=b[i];
	}
	for(int j=1;(1<<j)<=n;j++)
	{
		for(int i=1;i+(1<<j)-1<=n;i++)
		{
			dp2[i][j]=max(dp2[i][j-1],dp2[i+(1<<j-1)][j-1]);
		}
	}
	scanf("%llu",&q);
	for(int i=1;i<=q;i++)
	{
		scanf("%llu %llu",&l,&r);
		query[r].push_back(make_pair(l,i));
	}
	Build(1,1,n);
	for(int i=1;i<=n;i++)
	{
		l=1;
		r=i;
		int key=i;
		while(l<=r)
		{
			int mid=(l+r)>>1;
			if(Query1(mid,i)==a[i])
			{
				key=mid;
				r=mid-1;
			}
			else
			{
				l=mid+1;
			}
		}
		Update(1,key,i,Make(a[i],0,0,0,0,0));
	//	printf("%d ",key);
		l=1;
		r=i;
		key=i;
		while(l<=r)
		{
			int mid=(l+r)>>1;
			if(Query2(mid,i)==b[i])
			{
				key=mid;
				r=mid-1;
			}
			else
			{
				l=mid+1;
			}
		}
		Update(1,key,i,Make(0,b[i],0,0,0,0));
		Update(1,1,i,Make(0,0,0,0,0,1));
		//printf("%d\n",key);
		for(int j=0;j<query[i].size();j++)
		{
			pair<int,int>efc=query[i][j];
			Ans[efc.second]=Query(1,efc.first,i);
			twice=1;
			if(Query(1,efc.first,i)!=Query(1,efc.first,i))
			{
				printf(">fgg\n");
			}
			twice=0;
		}
	}
	for(int i=1;i<=q;i++)
	{
		printf("%llu\n",Ans[i]);
	}
} 

标签:lazy,比赛,int,Tree,long,NOIP2022,clox,cloy
From: https://www.cnblogs.com/kid-magic/p/16977218.html

相关文章

  • 【游记】NOIp2022 游记
    开头b话上次发游记还是上次。上次发游记还是上次NOIp,后来,可能是因为自己常常对比赛中的表现并不满意,所以一次也没写过。这次也不算满意吧,但是我意识到,再不写的话,可能......
  • NOIP2022 题解
    A.种花枚举\((x_2,y_0)\),考虑\(x_1\)可能在哪些位置,显然是在\(x_2\)往上的一个极长连续0段上。考虑如果固定了\(x_1\)的位置后怎么计算C形的数量,我们预处理......
  • NOIP2022 总结
    \(\text{summary}\)怎么都没想到这次题目那么有新意:把这样的题\(T2\)放\(T2\)......策略出现很大问题,赛后也意识到很多选手也会出现同样的问题:死磕\(T2\)毕竟稳妥的......
  • [NOIP2022] 喵了个喵 题解
    [NOIP2022]喵了个喵题解先考虑\(k=2n-2\),这个数字提示我们每个栈放两种颜色,剩下一个栈辅助操作。那么颜色被分为两类在栈底,可以通过操作2消去。在栈顶,可以通过操作1......
  • P8865 [NOIP2022] 种花
    简要题意\(T\)组数据,给你一个\(n\timesm\)的\(01\)矩阵。\(0\)部分可以组成\(A_c\)个\(\texttt{C}\)型图案和\(A_f\)个\(\texttt{F}\)型图案。你需要输出......
  • 【题解】P8866 [NOIP2022] 喵了个喵(构造,adhoc)
    【题解】P8866[NOIP2022]喵了个喵题目链接P8866[NOIP2022]喵了个喵题意概述有一个牌堆和\(n\)个可以从栈底删除元素的栈,任务是要通过规则将所有的卡牌消去。开......
  • P8867 NOIP2022 建造军营
    P8867NOIP2022建造军营-洛谷|计算机科学教育新生态(luogu.com.cn)。给定一个无向联通图\(G=(V',E')\),求有多少个二元组\((V,E)\),满足:\(V\subseteqV'\),\(......
  • 某行业比赛部分WEB题解题思路
    login.php源码如下:<?phperror_reporting(0);highlight_file(__FILE__);include_once("flag.php");$data=['username'=>'admin','password'=......
  • NOIP2022 T3 建造军营
    只有B国炸毁了图的割边,才会使得图不连通,进而可能会导致军营不连通。也就是说,A国可以随意地看守或不看守不是割边的边。因此想到边双缩点后树形DP。为什么边双缩点后会......
  • NOIP2022 总结
    一定要先把可能写出的正解写好了再去打别的暴力(时间不足导致T3建造军营推出式子但没时间写\(100\to0\))。特殊的多测不清空(T1种花未清空读入时的字符串数组\(100\t......