首页 > 其他分享 >Luogu4315 月下“毛景树” - 树链剖分 -

Luogu4315 月下“毛景树” - 树链剖分 -

时间:2022-11-02 20:57:26浏览次数:83  
标签:dfn 剖分 tx 毛景树 mid int num maxn Luogu4315

题目链接:https://www.luogu.com.cn/problem/P4315

题意:一棵有边权的树,维护树上的链加、链覆盖、修改边权、链上max

题解:
好难写...
首先把边权转化为儿子的点权
然后树链剖分,需要注意覆盖和加的先后顺序。可以这么考虑:
有一串操作,为 加 覆盖 加 覆盖 加 覆盖...,应该怎么打懒标记?
显然,如果要打覆盖标记,前面的加和覆盖标记都要清除,而要打加标记就直接打
然后pushdown的时候就先处理覆盖标记再处理加标记
注意单点修改的时候别忘了pushdown!这样才能保证当前的操作是最新版本的(不会被之前的懒标记pushdown所影响)
一个细节:由于边权转点权了,所以最后跳到一条重链上之后需要更新/查询的是(dfn[tx]+1,dfn[ty]),(设tx是ty的祖先)因为tx点权对应的是tx到tx父亲的边的边权

// by SkyRainWind
#include <cstdio>
#include <vector>
#include <cstring>
#include <iostream>
#include <algorithm>
#define mpr make_pair
#define debug() cerr<<"Yoshino\n"
#define rep(i,a,b) for(int (i)=(a);(i)<=(b);(i)++)
#define pii pair<int,int>

using namespace std;

typedef long long LL;

const int inf = 1e9, INF = 0x3f3f3f3f, maxn = 2e5+5;

int n;
vector<int>g[maxn];
int val[maxn];
int dfn[maxn], seq[maxn], dfs_clock;
int dep[maxn], heavy[maxn], top[maxn], sz[maxn], fa[maxn];
int X[maxn], Y[maxn], Z[maxn];
struct seg{
	int mx, lazyadd, lazyall;
}se[maxn << 2];

void dfs1(int x,int fat){
	fa[x] = fat;
	sz[x] = 1;
	dep[x] = dep[fat] + 1;
	
	for(int u : g[x]){
		if(u == fat)continue;
		dfs1(u, x);
		sz[x] += sz[u];
		if(!heavy[x] || sz[u] > sz[heavy[x]])
			heavy[x] = u;
	}
}

void dfs2(int x,int now){
	top[x] = now;
	dfn[x] = ++ dfs_clock;
	seq[dfs_clock] = x;
	
	if(!heavy[x])return ;
	dfs2(heavy[x], now);
	for(int u : g[x]){
		if(u == heavy[x] || u == fa[x])continue;
		dfs2(u,u);
	}
}

void build(int x,int y,int num){
	if(x == y){
		se[num].lazyadd = 0;se[num].lazyall = -1;
		se[num].mx = val[seq[x]];
		return ;
	}
	int mid = x + y >> 1;
	build(x,mid,num << 1);build(mid+1, y, num<<1 | 1);
	se[num].mx = max(se[num<<1].mx, se[num << 1|1].mx);
	se[num].lazyadd = 0;se[num].lazyall = -1;
}

void pushdown(int l,int r,int num){
	if(~se[num].lazyall){
		se[num << 1].mx = se[num<<1 | 1].mx = se[num].lazyall;
		se[num << 1].lazyall = se[num << 1 | 1].lazyall = se[num].lazyall;
		se[num << 1].lazyadd = se[num << 1 | 1].lazyadd = 0;
		se[num].lazyall = -1;
	}
	if(se[num].lazyadd){
		se[num << 1].mx += se[num].lazyadd;
		se[num << 1|1].mx += se[num].lazyadd;
		se[num << 1].lazyadd += se[num].lazyadd;
		se[num << 1|1].lazyadd += se[num].lazyadd;
		se[num].lazyadd = 0;
	}
}

void add(int x,int y,int k,int l,int r,int num){
	if(x <= l && r <= y){
		se[num].mx += k;
		se[num].lazyadd += k;
		return ;
	}
	pushdown(l,r,num);
	int mid = l+r >> 1;
	if(y <= mid)add(x,y,k,l,mid,num<<1);
	else if(x > mid)add(x,y,k,mid+1,r,num<<1|1);
	else add(x,y,k,l,mid,num<<1), add(x,y,k,mid+1,r,num<<1|1);
	se[num].mx = max(se[num << 1].mx, se[num << 1| 1].mx);
}

void cover(int x,int y,int k,int l,int r,int num){
	if(x <= l && r <= y){
		se[num].mx = k;
		se[num].lazyadd = 0;
		se[num].lazyall = k;
		return ;
	}
	pushdown(l,r,num);
	int mid = l+r >> 1;
	if(y <= mid)cover(x,y,k,l,mid,num<<1);
	else if(x > mid)cover(x,y,k,mid+1,r,num<<1|1);
	else cover(x,y,k,l,mid,num<<1), cover(x,y,k,mid+1,r,num<<1|1);
	se[num].mx = max(se[num << 1].mx, se[num << 1| 1].mx);
}

int query(int x,int y,int l,int r,int num){
	if(x <= l && r <= y)
		return se[num].mx;
	pushdown(l, r, num);
	int mid = l+r>>1;
	if(y<=mid)return query(x,y,l,mid,num<<1);
	else if(x>mid)return query(x,y,mid+1,r,num<<1|1);
	else return max(query(x,y,l,mid,num<<1),query(x,y,mid+1,r,num<<1|1));
}

void chain_add(int x,int y,int k){
	int tx = top[x], ty = top[y];
	while(tx != ty){
		if(dep[tx] < dep[ty])swap(x, y), swap(tx, ty);
		add(dfn[tx], dfn[x], k, 1, n ,1);
		x = fa[tx], tx = top[x]; 
	}
	if(dep[x] > dep[y])swap(x, y);
	if(x!=y)add(dfn[x]+1, dfn[y], k, 1, n, 1);
}

void chain_cover(int x,int y,int k){
	int tx = top[x], ty = top[y];
	while(tx != ty){
		if(dep[tx] < dep[ty])swap(x, y), swap(tx, ty);
		cover(dfn[tx], dfn[x], k, 1, n ,1);
		x = fa[tx], tx = top[x]; 
	}
	if(dep[x] > dep[y])swap(x, y);
	if(x!=y)cover(dfn[x]+1, dfn[y], k, 1, n, 1);
}

void modify(int tx,int t,int l,int r,int num){
	if(l == r){
		se[num].mx = t;
		return ;
	}
	pushdown(l,r,num);
	int mid = l+r>>1;
	if(tx<=mid)modify(tx,t,l,mid,num<<1);
	else modify(tx,t,mid+1,r,num<<1|1);
	se[num].mx = max(se[num << 1].mx, se[num<<1|1].mx);
}

int chain_max(int x,int y){
	int tx = top[x], ty = top[y];
	int sum = -1e9;
	while(tx != ty){
		if(dep[tx] < dep[ty])swap(x, y), swap(tx, ty);
//		cerr << tx << " " << x << " " << dfn[tx] << " " << dfn[x] << "\n";
//		printf("?? %d %d %d %d\n",x,tx,dfn[tx]+1,dfn[x]);
		sum = max(sum, query(dfn[tx], dfn[x], 1, n, 1));
		x = fa[tx], tx = top[x];
	}
	if(dep[x] > dep[y])swap(x, y);
//	cerr << dfn[x] +1 << " " << dfn[y] << "\n";
	if(x != y)sum = max(sum, query(dfn[x] + 1, dfn[y], 1, n,1));
	return sum;
}

signed main(){
//	freopen("4315.in","r",stdin);
//	freopen("P4315.out","w",stdout);
	scanf("%d",&n);
	for(int i=1;i<=n-1;i++){
		scanf("%d%d",&X[i],&Y[i]);scanf("%d",&Z[i]);
		g[X[i]].push_back(Y[i]), g[Y[i]].push_back(X[i]);
	}
	dfs1(1,0);
	dfs2(1,1);
	for(int i=1;i<=n-1;i++){
		int tx = X[i], ty = Y[i];
		if(fa[tx] != ty)swap(tx, ty);
		val[tx] = Z[i]; 
	}
	build(1,n,1);
//	printf("?? %d %d\n",top[4],top[1]);
	while(1){
		char op[12];scanf("%s",op + 1);
		if(op[1] == 'S')break;
		if(op[1] == 'A'){	// Add
			int x,y,z;scanf("%d%d%d",&x,&y,&z);
			chain_add(x,y,z);
		}
		if(op[1] == 'M'){	// Max
			int x,y;scanf("%d%d",&x,&y);
			printf("%d\n",chain_max(x, y));
		}
		if(op[2] == 'h'){	// Change
			int x,z;scanf("%d%d",&x,&z);
			int tx = X[x], ty = Y[x];
			if(dep[tx] < dep[ty])swap(tx, ty);
			modify(dfn[tx],z,1,n,1);
		}
		if(op[2] == 'o'){	// Cover
			int x,y,z;scanf("%d%d%d",&x,&y,&z);
			chain_cover(x,y,z);
		}
	}

	return 0;
}

标签:dfn,剖分,tx,毛景树,mid,int,num,maxn,Luogu4315
From: https://www.cnblogs.com/SkyRainWind/p/16852398.html

相关文章

  • CF487E Tourists(Tarjan,圆方树,树链剖分,线段树)
    CF487ETourists带权无向图\(N,M\),\(Q\)次询问\(s,t\)所有不经过重复点可能路径经过的最小值的最小值。CODE每次修改一个圆点\(u\)周围的方点有点难。可是李神......
  • 【十二省联考2019】希望(容斥,换根DP,长链剖分,懒标记)
    暴力的想法是考虑钦定每个点作为到达点并统计贡献,但显然这样会算重。注意到会算重的到达点一定构成了一个连通块,这是一个很好的性质,方便了我们容斥:我们直接用点的贡献减去......
  • 【XSY3726】大猫熊和他的k边形(三角剖分,卡特兰数)
    记录一下两个结论。有标号\(n\)边形的三角剖分数等于\(B_{n-2}\),其中\(B\)是卡特兰数。证明:考虑DP,设\(C_n\)为有标号\(n\)边形的三角剖分数,考虑与\(1\)号......
  • 【XSY3679】农民(树链剖分)
    题面题解先考虑一个节点怎么样才会被走到。对于一个权值为为\(x\)的节点,它的左子树内的节点有可能被走到仅当其权值小于\(x\),右子树内的节点有可能被走到仅当其......
  • 【XSY3633】匹配(树形 DP,树链剖分,分治)
    考虑最普通的DP:设\(f_{u,i,0/1}\)表示\(u\)子树内恰好包含\(i\)条边的最大权匹配,其中\(u\)有无被匹配。这是个树上背包,暴力DP是\(O(n^2)\)的。可以发现\(f......
  • 【WC2010】重建计划(分数规划+长链剖分)
    长链剖分因为有很多巨佬只是讲了一下大致的做法,并没有详细地解释如何维护,所以就有了这篇题解。巨佬们都不屑于详细写,我太弱了/kk首先先对原树进行长链剖分。先讲一些定......
  • 【bzoj2402】陶陶的难题II(分数规划+树链剖分+斜率优化+半平面交)
    题目让我们维护这么一个东西:\(\dfrac{y_i+q_j}{x_i+p_j}\)的最大值。容易想到分数规划,二分枚举答案\(mid\),则有:\(\dfrac{y_i+q_j}{x_i+p_j}=mid\)化简:\(y_i+q_j=mid\t......
  • 树链剖分
    剖分过程voiddfs1(intu){ son[u]=-1; siz[u]=1; for(inti=hd[u];i;i=g[i].nxt) { intv=g[i].to; if(!dep[v]) { dep[v]=dep[u]+1; ......
  • 【图论】长链剖分学习笔记
    参考文章1参考文章20x01:引入与重链剖分不同,长链剖分以子树深度最大的儿子作为重儿子,这里所述之深度是指子树内离它最远的叶子到它的距离。如图绿色部分就是长链。......
  • 【转载】毛毛虫剖分
    转载声明转载自关于毛毛虫剖分-一粒夸克定义一种由重链剖分推广而成的树上结点重标号方法,支持修改/查询一只毛毛虫的信息,并且可以对毛毛虫的身体和足分别修改/查询......