题解 LOJ2049 「HNOI2016」网络

题目链接

简要题意:给定一棵树。让你支持三种操作:

  1. 加入一条路径\((u,v)\)。路径有一个权值\(w\)
  2. 删除之前某一时刻加入的路径。
  3. 对树上某个节点\(u\)询问:问当前所有不经过\(u\)的路径的权值的最大值。

可以发现,本题的难点在于最大值,它不像类似于“求权值和”这样的问题,它不支持撤销,也就是没有“可减性”。因此我们很难通过一般的数据结构,完成删除操作,并维护最大值。

那么换个思路:既然不能直接维护答案,考虑对于每次询问都二分答案。二分之后,求最大值的问题,就转化为了数路径数量的问题。具体来说,设二分值为\(\text{mid}\)。那么考虑所有权值大于\(\text{mid}\)的路径。如果这些路径全部覆盖了询问点\(u\),则该询问的答案小于等于\(\text{mid}\),否则该询问的答案大于\(\text{mid}\)

加入、删除一条路径。以加入为例。可以令路径的两个端点点权\(+1\),LCA和LCA的父亲点权\(-1\),则经过某个节点\(u\)的路径数量,就是\(u\)子树内所有节点的点权和。求出dfs序后,问题转化为单点修改、区间求和。可以用树状数组实现,常数很小。

如果对每次询问都二分答案,然后暴力加入这些路径,则单次询问的时间复杂度达到\(O(n\log^2n)\),总时间复杂度\(O(mn\log^2n)\),无法承受。

注意到,对于任何一个询问,假设二分值为\(\text{mid}\),要考虑的都是权值大于\(\text{mid}\)的这些路径。也就是说,只要二分值相同,对于不同的询问,我们在树状数组上加入的是同一些路径!于是我们考虑把这样的加入路径的操作放在一起做。简单讲,我们之前的做法是:对每个询问,求出二分值,做一遍加入路径的操作;现在变成:对每个二分值,加入对应的路径,然后一次性处理所有对应的询问。这就是整体二分。

我们把它写成一个分治的过程。每次分治传入的参数是两个东西:(1) 当前二分的答案区间\([l,r]\),(2) 这个区间对应的操作。这些操作又分为两种,一是路径权值在\([l,r]\)之间的插入、删除操作,二是我们已经确定其答案在\([l,r]\)之间的询问操作。每次传入的这些操作在传入时是按时间顺序排好的。我们只要按顺序处理它们,然后将其分配到左半边、右半边去即可。

时间复杂度\(O(m\log^2n)\)

参考代码(在LOJ查看):

//problem:LOJ2049
#include <bits/stdc++.h>
using namespace std;

#define pb push_back
#define mk make_pair
#define lob lower_bound
#define upb upper_bound
#define fi first
#define se second
#define SZ(x) ((int)(x).size())

typedef unsigned int uint;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pii;

const int MAXN=1e5,MAXM=2e5;
int n,m;
struct EDGE{int nxt,to;}edge[MAXN*2+5];
int head[MAXN+5],tot;
inline void add_edge(int u,int v){edge[++tot].nxt=head[u],edge[tot].to=v,head[u]=tot;}
int fa[MAXN+5],sz[MAXN+5],son[MAXN+5],dep[MAXN+5];
void dfs1(int u){
	sz[u]=1;
	for(int i=head[u];i;i=edge[i].nxt){
		int v=edge[i].to;
		if(v==fa[u])continue;
		fa[v]=u;
		dep[v]=dep[u]+1;
		dfs1(v);
		sz[u]+=sz[v];
		if(!son[u]||sz[v]>sz[son[u]])son[u]=v;
	}
}
int top[MAXN+5],dfn[MAXN+5],ofn[MAXN+5],rev[MAXN+5],cnt_dfn;
void dfs2(int u,int t){
	top[u]=t;
	dfn[u]=++cnt_dfn;
	rev[cnt_dfn]=u;
	if(son[u])dfs2(son[u],t);
	for(int i=head[u];i;i=edge[i].nxt){
		int v=edge[i].to;
		if(v==fa[u]||v==son[u])continue;
		dfs2(v,v);
	}
	ofn[u]=cnt_dfn;
}
int get_lca(int u,int v){
	while(top[u]!=top[v]){
		if(dep[top[u]]<dep[top[v]])swap(u,v);
		u=fa[top[u]];
	}
	return (dep[u]<dep[v])?u:v;
}

struct FenwickTree{
	int c[MAXN+5];
	void modify(int p,int x){
		for(;p<=n;p+=(p&(-p)))c[p]+=x;
	}
	int query(int p){
		int res=0;
		for(;p;p-=(p&(-p)))res+=c[p];
		return res;
	}
	int query(int l,int r){
		return query(r)-query(l-1);
	}
	FenwickTree(){}
}T;

void path_modify(int u,int v,int x){
	T.modify(dfn[u],x);
	T.modify(dfn[v],x);
	int lca=get_lca(u,v);
	T.modify(dfn[lca],-x);
	if(fa[lca])T.modify(dfn[fa[lca]],-x);
}

int ans[MAXM+5];
struct Query_t{
	int op,u,v,w,id;
}q[MAXM+5];
void solve(int ans_l,int ans_r,int pos_l,int pos_r){
	if(ans_l==ans_r){
		for(int i=pos_l;i<=pos_r;++i)if(q[i].op==2)ans[q[i].id]=ans_l;
		return;
	}
	static Query_t que_l[MAXM+5],que_r[MAXM+5];
	int mid=(ans_l+ans_r)>>1;
	int cnt_l=0,cnt_r=0,sum=0;
	bool have_query_l=false,have_query_r=false;
	for(int i=pos_l;i<=pos_r;++i){
		if(q[i].op==2){
			if(T.query(dfn[q[i].u],ofn[q[i].u])==sum){
				que_l[++cnt_l]=q[i];
				have_query_l=true;
			}
			else{
				que_r[++cnt_r]=q[i];
				have_query_r=true;
			}
		}
		else{
			if(q[i].w<=mid){
				que_l[++cnt_l]=q[i];
			}
			else{
				int x=(q[i].op==0?1:-1);
				path_modify(q[i].u,q[i].v,x);
				sum+=x;
				que_r[++cnt_r]=q[i];
			}
		}
	}
	for(int i=pos_l;i<=pos_r;++i){
		if(q[i].op!=2&&q[i].w>mid){
			int x=(q[i].op==0?1:-1);
			path_modify(q[i].u,q[i].v,-x);
		}
	}//树状数组清空
	for(int i=1;i<=cnt_l;++i)q[pos_l+i-1]=que_l[i];
	for(int i=1;i<=cnt_r;++i)q[pos_l+cnt_l+i-1]=que_r[i];
	if(have_query_l)solve(ans_l,mid,pos_l,pos_l+cnt_l-1);
	if(have_query_r)solve(mid+1,ans_r,pos_l+cnt_l,pos_r);
}

int main() {
	cin>>n>>m;
	for(int i=1,u,v;i<n;++i)cin>>u>>v,add_edge(u,v),add_edge(v,u);
	dfs1(1);dfs2(1,1);
	int max_w=0;
	for(int i=1;i<=m;++i){
		cin>>q[i].op;
		q[i].id=i;
		if(q[i].op==0){
			cin>>q[i].u>>q[i].v>>q[i].w;
			max_w=max(max_w,q[i].w);
			ans[i]=-3;
		}
		else if(q[i].op==1){
			int t;cin>>t;
			q[i].u=q[t].u;
			q[i].v=q[t].v;
			q[i].w=q[t].w;
			ans[i]=-3;
		}
		else{
			cin>>q[i].u;
		}
	}
	solve(-1,max_w,1,m);
	for(int i=1;i<=m;++i)if(ans[i]!=-3)cout<<ans[i]<<endl;
	return 0;
}
posted @ 2020-05-15 10:58  duyiblue  阅读(178)  评论(0编辑  收藏  举报