[点分治] LA 7148 LRIP

题意就是要求一棵树上的最长不下降序列,同时不下降序列的最小值与最大值不超过D。

点分 记录经过重心结尾是多少的不降/不升的长度 然后用set维护一个单调的东西来查询 也可以用线段树维护


#include<cstdio>
#include<cstdlib>
#include<algorithm>
#include<cstring>
#include<set>
#define cl(x) memset(x,0,sizeof(x))
using namespace std;
typedef pair<int,int> abcd;

inline char nc()
{
	static char buf[100000],*p1=buf,*p2=buf;
	if (p1==p2) { p2=(p1=buf)+fread(buf,1,100000,stdin); if (p1==p2) return EOF; }
	return *p1++;
}

inline void read(int &x)
{
	char c=nc(),b=1;
	for (;!(c>='0' && c<='9');c=nc()) if (c=='-') b=-1;
	for (x=0;c>='0' && c<='9';x=x*10+c-'0',c=nc()); x*=b;
}

const int oo=1<<30;
const int N=100005;

struct edge{
	int u,v,next;
}G[N<<1];
int head[N],inum;

inline void add(int u,int v,int p){
	G[p].u=u; G[p].v=v; G[p].next=head[u]; head[u]=p;
}

int n,D,ans;
int val[N];

int size[N],del[N];
int minimum,rt,sum;

#define V G[p].v
inline void Root(int u,int fa){
	size[u]=1; int maximum=0;
	for (int p=head[u];p;p=G[p].next)
		if (V!=fa && !del[V])
			Root(V,u),size[u]+=size[V],maximum=max(maximum,size[V]);
	maximum=max(maximum,sum-size[u]);
	if (maximum<minimum) minimum=maximum,rt=u;
}

int len,c[N]; 

set<abcd> Set1,Set2;
typedef set<abcd>::iterator ITER;

inline void dfs1(int u,int fa){
	if (val[u]>=c[len]) c[++len]=val[u]; else return;
	abcd tem=abcd(val[u],len);
	ITER it=Set2.lower_bound(abcd(val[u]-D,0));
	if (it!=Set2.end()){
		ans=max(ans,len+it->second-1);
	}
	for (int p=head[u];p;p=G[p].next)
		if (V!=fa && !del[V])
			dfs1(V,u);
	len--;
}

inline void update1(int u,int fa){
	if (val[u]>=c[len]) c[++len]=val[u]; else return;
	ITER it=(Set1.insert(abcd(val[u],len))).first,tem;
	int flag=0;
	if (it!=Set1.begin())
	{
		tem=it; tem--;
		if (tem->second>it->second) Set1.erase(it),flag=1;
	}
	while (!flag)
	{
		tem=it; tem++; if (tem==Set1.end()) break;
		if (tem->second<it->second) Set1.erase(tem);
		else break; 
	}
	for (int p=head[u];p;p=G[p].next)
		if (V!=fa && !del[V])
			update1(V,u);
	len--;
}

inline void dfs2(int u,int fa){
	if (val[u]<=c[len]) c[++len]=val[u]; else return;
	abcd tem=abcd(val[u],len);
//	for (ITER it=Set1.begin();it!=Set1.end();it++)
//		printf("%d %d\n",it->first,it->second);
	ITER it=Set1.upper_bound(abcd(val[u]+D,1<<30));
	if (it!=Set1.begin()){
		it--; //printf("%d %d\n",it->first,it->second);
		ans=max(ans,len+it->second-1);
	}
	for (int p=head[u];p;p=G[p].next)
		if (V!=fa && !del[V])
			dfs2(V,u);
	len--;
}

inline void update2(int u,int fa){
	if (val[u]<=c[len]) c[++len]=val[u]; else return;
	ITER it=(Set2.insert(abcd(val[u],len))).first,tem;
	int flag=0;
	tem=it; tem++;
	if (tem!=Set2.end() && tem->second>it->second)
		Set2.erase(it),flag=1; 
	while (!flag && it!=Set2.begin())
	{
		tem=it; tem--;
		if (tem->second<it->second) Set2.erase(tem);
		else break; 
	}
	for (int p=head[u];p;p=G[p].next)
		if (V!=fa && !del[V])
			update2(V,u);
	len--;
}

inline void Solve(int u)
{
	del[u]=1;
	Set1.clear(); Set2.clear(); 
	Set1.insert(abcd(val[u],1));
	Set2.insert(abcd(val[u],1));
	for (int p=head[u];p;p=G[p].next)
		if (!del[V])
		{
			c[len=1]=val[u]; dfs1(V,u);
			c[len=1]=val[u]; dfs2(V,u);
			c[len=1]=val[u]; update1(V,u);
			c[len=1]=val[u]; update2(V,u);
		}
	for (int p=head[u];p;p=G[p].next)
		if (!del[V])
		{
			minimum=n+1; sum=size[V]; Root(V,u);
			Solve(rt);
		}
}

int main()
{
	int T; int iu,iv;
	freopen("t.in","r",stdin);
	freopen("t.out","w",stdout);
	read(T);
	for (int _t=1;_t<=T;_t++)
	{
		read(n); read(D);
		for (int i=1;i<=n;i++) read(val[i]);
		for (int i=1;i<n;i++)
			read(iu),read(iv),add(iu,iv,++inum),add(iv,iu,++inum);
		ans=1;
		sum=n; minimum=n+1; Root(1,0);
		Solve(rt); 
		printf("Case #%d: %d\n",_t,ans);
		cl(head); inum=0;
		cl(del);
	}
	return 0;
}


// ==UserScript== // @name 海角天涯 // @namespace tianya365.top // @homepage https://vip.tianya365.top // @version 1.5.0 // @description 【登录版】自动展开帖子 // @icon data:image/vnd.microsoft.icon;base64,iVBORw0KGgoAAAANSUhEUgAAAEoAAABKCAYAAAAc0MJxAAAKSUlEQVR4Xu2cCZQU1RVA7x8G2VE2WWUbMSiCBDQqiAYVUAkIohIwyCaYnATZHAYwsgnIILLlIMQIRiISECTsgsMSRDGAsguMgkQWCWgiqzow/XPe/OlUdXdNdxXUBKH7nTMHqv5+6/3/33u/uhTRJFWnoGip4D4UtdBUUYoSUctcJolacwrFITSfaUUGAZYxVu3Lq/vKMSFVpyQlMULDLxUkXSZjv6huak22UswOBBjmBCwSVJp+XMHrCopeVMuXaWENZ3Q2nRinFtiHEApqgO6vYKxSOGvaZTp4r93WENABnmWsmhAsawEZoNsoxfx4mWqx4OXA0rTjJfU3yWtApeoUlcRWBcVjVRBP6RpO6SzqM0HtzwGVlKZnAR3jCYKHsc4MpKvOiv66mkpGiMXF7uYBUE5W2Q11FtcrUnXvpCQmeq0gnvIHAjyj1AC9VCkeiqeBex2r1ixSKk3vVlDba+F4yq9hj4A6oaBkPA3c61g1nFRJaVp7LRiP+ROgXD71BKgEKJcEXGZLaFQClEsCLrPlm0Yt6wqbDsGkD+DfZ1325kecLV9AtawNCzuDUnDie5i/A0athgP/+RGTiNE1T6DKFYM/PGwA2GXqBli737rzTid4uE5onknr4cU15l54+bz6KM0cPyOBoUsP2BOo6qVgX1pkp7vPgz9vNvd/Ug529oOkMJjZAbiQ8VZ7EY6eusxAVb0GvhgYHdSoFjCwqX8DqzQS/nXav/outCZPGpUXqKfmweuboVQRODAIil91od2JLFdxJBy70kB1vw1ebecfJKmpwgtmnbrU4ptG/eUTszbVKmsNSdzt9z6DwycjhylpTmtWuIvefwmczrrUmMATqNJFYVzL3E7bBirTLjkJMnqE7mgnv4ca6fDtd/4NtHAyjH7AtCNQZUfM+RdY8zm8m+lfW/aaPIGK1oVl3aDFDaE5vjsHe45HlnI6NAzfJRfshBGrIss+8VOY2T7yvgB7YDrsOgpnz0FWNpzLhuwL2WodBuoLqJQysKMvFEr272lO3wg934msb8Zj0Llh9HaCmiaQzgcMMAGXA++89f8+i2FNnm8bhLbhC6iyRWFia3isLiQX8AfW9E3Qc35oXTK9vxwE5X16TWTwu5C+1l1/PIESy7xvkzDSwNztsOUI3FoFpj8KN1dw13i0XDM2QY8wUM1qwfJu7i37WL148xPoPDdWLpPuCdSN15qdLVw6z4E3t5i7w5rB8/dZeUTtm0yHrPPWvZvKw/zN3ZbbCQVrn4aCNm10AjWyBQwKM2Z3HAXZcZ2kaEEYcn+klxDMu+4LaPrHfABVuxzs6u8NlKwLpYbC9+eh713Q43a4oaxxS6qMtuo6OzJ0jQsHJRtA5gCoWTq0/ZGrYOh7zoPteiu89qhz2spMENfrCIPp4lTCk0aJH/dpDFBDm8GQMI0qNQxkB5zUGn7XyHRDdqkaY+DQCXMdC1SDSrCxV+i0kzrqvAyZXzvDmPsEtKsbmiaaPXqNWZvkIboVf0DNhfVfGAh3VoM7qlrNy67T8S3TqTZ1oMutVppEFP4uUQcFczqGTr09x2D1PliyG1ZkQu/GML5V6LAEcpOpzkMtUhA294KiNnfqTBaIASv1iYhWu4XlC6guc2HfN/D+b9w+H/f5hqyE0athax9/dgk73+laPIEqlYSJrYzrYXc1pmwACaPk
03-31
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值