2019 ICPC Asia Xuzhou Regional. H. Yuuki and a problem(树状数组套主席树)

2021-03-12 06:27

阅读:655

标签:efi   strong   tps   des   include   back   mat   pow   bit   

题目链接


\(Description\)
给定长为\(n\)的序列\(A_i\),两种操作:

  1. 将某个数\(A_i\)修改为\(v\)
  2. 查询用区间\([l,r]\)内的数不能组成的最小的数(能组成\(v\)是指存在一个\([l,r]\)的子集\(s\)使\(s\)的和等于\(v\))。
    \(n,A_i\leq 2\times10^5\)

\(Solution\)
BZOJ(CodeChef)原题树套树版?技术图片

先考虑询问。设区间\([l,r]\)内已经可以组成的数为\([1,v]\)\([l,r]\)中最小的大于\(v\)的数为\(mx\),若\(mx>v+1\)则区间的答案即为\(v+1\);否则\(mx=v+1\)\(v\)可以更新为\(Sum(1,v+1)\)\([l,r]\)中值在\([1,v+1]\)的所有数的和),继续重复上述过程。
\(mx\)的时候可以求\(Sum(1,v+1)\),若\(Sum=mx\)则显然\(mx>v+1\);否则\(mx=v+1\)
容易发现\(v\)每次至少增加\(v+1\),且这个过程可以用主席树实现。所以查询复杂度为\(O(\log V\log n)\)
对于修改,改成带修改主席树(树状数组套主席树)就可以了。复杂度\(O(n\log V\log^2n)\)

这个带修改主席树,每个位置维护一个前缀和即可,查询是单点查询。(不太懂他们麻烦的写法技术图片


//1144ms	486.4MB
#include 
#define pc putchar
#define gc() getchar()
typedef long long LL;
const int N=2e5+5,V=2e5;

int A[N];
struct President_Tree
{
	#define S N*155//N*18*18*2 //注意MLE。。
	#define ls son[x][0]
	#define rs son[x][1]
	#define lson ls,l,m
	#define rson rs,m+1,r
	int tot,son[S][2];
	LL pre[S];
	#define R(x) (rs?rs:(rs=++tot))
	void Insert(int &x,int l,int r,int p)
	{
		!x&&(x=++tot);
		if(l==r) {pre[x]+=l; return;}
		int m=l+r>>1; p>1; p>1; return pre[x]+(p vl,vr;
	#define lb(x) (x&-(x))
	void Modify(int p,int a,int v)
	{
		for(; p

2019 ICPC Asia Xuzhou Regional. H. Yuuki and a problem(树状数组套主席树)

标签:efi   strong   tps   des   include   back   mat   pow   bit   

原文地址:https://www.cnblogs.com/SovietPower/p/14091362.html


评论


亲,登录后才可以留言!