首页 > 其他分享 >Top Cluster 树分块入门学习笔记

Top Cluster 树分块入门学习笔记

时间:2024-12-09 15:32:30浏览次数:6  
标签:ft 分块 Top Fa Cluster maxn 树簇 ll 界点

定义

  • 树簇(Cluster):将树上的边划分为若干个连通块,称为树簇。

  • 界点、内点:每个树簇内有两个界点,其他点为内点,满足两个树簇至多交于一个界点。

  • 簇路径:对于每个树簇,其内部两个界点之间的路径为簇路径。

由于这里不是学习 Top Tree 的地方,所以舍去了某些其他内容。

树簇分块

给定一个常数 \(B\),现在将树划分为若干个树簇,满足每个树簇大小 \(\le B\),并且树簇个数为 \(\mathcal O(B)\)。

下面介绍 Top Cluster 树分块算法,首先,我们规定每个树簇中两个界点为祖先 - 子孙关系。

先 dfs 整棵树,并且开一个栈,维护当前还未被划分进树簇的边(一条边 \((x, fa_x)\) 可以用一个点 \(x\) 来表示)。

根据上面的规定,根节点是其所在簇的界点。对于当前点 \(u\),若满足以下任意一个条件,则将其标记为界点:

  • \(u\) 子树内与 \(u\) 在同一树簇内的界点数量超过 \(1\)。

  • \(u\) 子树内与 \(u\) 在同意树簇内的未被划分的边数 \(\ge B\)。

  • \(u\) 是根节点。

如果 \(u\) 被标记为了界点,此时我们需要将子树内与 \(u\) 连通的边划分为若干个树簇。根据以上分析,我们需要对于每个点 \(u\) 维护两个值 \(siz_u\) 表示当前子树内与 \(u\) 连通的边数,以及 \(ft_u\) 表示当前子树内与 \(u\) 连通的下界点(不存在则 \(ft_u = 0\))。

我们考虑以下算法流程:

  • 遍历所有儿子,并维护 \(sz\) 表示当前簇大小,以及 \(ct\) 表示当前簇中下方的界点数量。对于当前儿子,若可以加入当前簇则贪心加入,并实时维护 \(sz\) 和 \(ct\)。

  • 若不能加入,则将之前的所有边划分为一个树簇。找出对应的下界点,并对于每个点求出 \(top_u\) 表示 \(u\) 到根的路径中最近的界点,以及 \(near_u\) 表示最近的簇路径上的点。

点击查看代码
void add_cluster(ll u, ll v) {
		if(v) {
			Fa[v] = u, block[++len] = v, vis[u] = true; // 这里本质上我们就是按照所有界点来给树分块
			for(ll x = v; x ^ u; x = fa[x]) vis[x] = true;
		}
		for(ll i = 1; i <= cur_top; i++) {
			ft[cur[i]] = v; cur[i] != v && (top[cur[i]] = u);
			if(!v) { near[cur[i]] = u; continue; }
			for(ll x = cur[i]; x; x = fa[x])
				if(vis[x]) { near[cur[i]] = x; break; }
				else if(near[x]) { near[cur[i]] = near[x]; break; }
		}
	}
void work(ll u, ll v) {
		cur_top = 0, ++tot; ll now = 0; // cur 为当前划分进树簇的边,tot 为簇的编号,now 为下界点
		do {
			cur[++cur_top] = stk[stk_top];
			now |= ft[stk[stk_top]], node[tot].pb(stk[stk_top]);
			ind[stk[stk_top]] = tot;
		} while(stk[stk_top--] != v);
		add_cluster(u, now); // 加入一个树簇,界点为 u 和 now
		Top[tot] = u, Foot[tot] = now;
	}

例题

Luogu6778

转化为求区间内任意两个点之间的 LCA 深度,莫队二次离线后转化为 \(\mathcal O(n)\) 次到根路径加法,以及查询一个点到根路径权值和。

先 Top Cluster 树分块,修改时,将一条路径分为:

  • 当前点到所在簇上界点的路径。

  • 上界点到根的路径。

类似于分块,维护 \(bsum_u\) 表示簇内 \(u\) 到达上界点的路径权值和,以及 \(sum_u\) 表示当前簇路径权值和(\(u\) 为当前簇下界点,若 \(u\) 不为界点定义无效),同时维护一个标记 \(tag_u\) 表示当前簇路径加的次数。

查询时,分为三部分:

  • 当前点 \(u\) 到 \(near_u\) 的路径。

  • \(near_u\) 到 \(top_u\) 的路径。

  • \(top_u\) 到根的路径。

时间复杂度 \(\mathcal O((n + m) \sqrt n)\)。

点击查看代码
#include <bits/stdc++.h>

namespace Initial {
	#define ll int
	#define ull unsigned int
	#define fi first
	#define se second
	#define mkp make_pair
	#define pir pair <ll, ll>
	#define pb emplace_back
	#define i128 __int128
	using namespace std;
	const ll maxn = 2e5 + 10, inf = 1e9, mod = 998244353;
	ll power(ll a, ll b = mod - 2) {
		ll s = 1;
		while(b) {
			if(b & 1) s = 1ll * s * a %mod;
			a = 1ll * a * a %mod, b >>= 1;
		} return s;
	}
	template <class T>
	const ll pls(const T x, const T y) { return x + y >= mod? x + y - mod : x + y; }
	template <class T>
	void add(T &x, const T y) { x = x + y >= mod? x + y - mod : x + y; }
	template <class T>
	void chkmax(T &x, const T y) { x = x < y? y : x; }
	template <class T>
	void chkmin(T &x, const T y) { x = x > y? y : x; }
} using namespace Initial;

namespace Read {
	char buf[1 << 22], *p1, *p2;
	#define getchar() (p1 == p2 && (p2 = (p1 = buf) + fread(buf, 1, (1 << 22) - 10, stdin), p1 == p2)? EOF : *p1++)
	template <class T>
	void rd(T &x) {
		char ch; bool neg = 0;
		while(!isdigit(ch = getchar()))
			if(ch == '-') neg = 1;
		x = ch - '0';
		while(isdigit(ch = getchar()))
			x = (x << 1) + (x << 3) + ch - '0';
		if(neg) x = -x;
	}
} using Read::rd;
ll B, n, m, bl[maxn]; ull ans[maxn], dis[maxn];
vector <pir> to[maxn];
struct query { ll l, r, id; } q[maxn];
vector <pir> vec1[maxn], vec2[maxn];
struct Data { ll l, r, w, id; }; vector <Data> vec[maxn];

namespace Top_Cluster {
	ll near[maxn], fa[maxn], siz[maxn], tot, ind[maxn];
	ll top[maxn], ft[maxn], stk[maxn], stk_top, cur[maxn], cur_top;
	ll Top[maxn], Foot[maxn], Fa[maxn]; bool vis[maxn];
	vector <ll> node[maxn]; ll block[maxn], len;
	
	void add_cluster(ll u, ll v) {
		if(v) {
			Fa[v] = u, block[++len] = v, vis[u] = true;
			for(ll x = v; x ^ u; x = fa[x]) vis[x] = true;
		}
		for(ll i = 1; i <= cur_top; i++) {
			ft[cur[i]] = v; cur[i] != v && (top[cur[i]] = u);
			if(!v) { near[cur[i]] = u; continue; }
			for(ll x = cur[i]; x; x = fa[x])
				if(vis[x]) { near[cur[i]] = x; break; }
				else if(near[x]) { near[cur[i]] = near[x]; break; }
		}
	}
	
	void work(ll u, ll v) {
		cur_top = 0, ++tot; ll now = 0;
		do {
			cur[++cur_top] = stk[stk_top];
			now |= ft[stk[stk_top]], node[tot].pb(stk[stk_top]);
			ind[stk[stk_top]] = tot;
		} while(stk[stk_top--] != v);
		add_cluster(u, now);
		Top[tot] = u, Foot[tot] = now;
	}
	void dfs(ll u, ll f = 0) {
		fa[u] = f;
		for(auto it = to[u].begin(); it != to[u].end(); it++)
			if(it -> fi == f) { to[u].erase(it); break; }
		siz[u] = 1, ft[u] = 0; ll cnt = 0;
		for(pir e: to[u]) {
			ll v = e.fi, w = e.se;
			dis[v] = dis[u] + w, stk[++stk_top] = v;
			dfs(v, u), ft[v] && (++cnt, ft[u] = ft[v]);
			siz[u] += siz[v];
		} ll now = 0;
		if(siz[u] >= B || cnt > 1 || u == 1) {
			ll sz = 0, ct = 0;
			for(ll i = to[u].size() - 1; ~i; i--) {
				ll v = to[u][i].fi, w = to[u][i].se;
				ct += ft[v] > 0, sz += siz[v];
				if(sz >= B || ct > 1) {
					sz = siz[v], ct = ft[v] > 0;
					work(u, to[u][i + 1].fi);
				}
			} work(ft[u] = top[u] = u, to[u][0].fi), siz[u] = 1;
		}
	}
} using namespace Top_Cluster;

ull tag[maxn], sum[maxn], bsum[maxn], dis_sum[maxn], f_ans[maxn];
void upd(ll u) {
	if(u == 1) return;
	for(ll i = 1; i < len; i++) sum[block[i]] -= sum[Fa[block[i]]];
	if(!Fa[u]) {
		sum[ft[u]] += dis[near[u]] - dis[top[u]];
		ll c = ind[u];
		for(ll v: node[c])
			if(!Fa[fa[v]]) bsum[v] -= bsum[fa[v]];
		for(; !Fa[u] && u > 1; u = fa[u]) bsum[u] += dis[u] - dis[fa[u]];
		for(ll i = node[c].size() - 1; ~i; i--) {
			ll v = node[c][i];
			if(!Fa[fa[v]]) bsum[v] += bsum[fa[v]];
		}
	}
	for(; u > 1; u = Fa[u]) ++tag[u], sum[u] += dis[u] - dis[Fa[u]];
	for(ll i = len - 1; i > 0; i--) sum[block[i]] += sum[Fa[block[i]]];
}
ull ask(ll u) {
	ull res = sum[top[u]];
	if(!Fa[u]) res += bsum[u];
	if(!Fa[u] && ft[u]) res += (dis[near[u]] - dis[top[u]]) * tag[ft[u]];
	return res;
}

int main() {
	rd(n), rd(m); B = max(1.2, n / sqrt(m));
	for(ll i = 1; i <= n; i++) bl[i] = (i - 1) / B + 1;
	for(ll i = 1, u, v, w; i < n; i++) {
		rd(u), rd(v), rd(w);
		to[u].pb(mkp(v, w)), to[v].pb(mkp(u, w));
	} dfs(1);
	for(ll i = 1; i <= n; i++) dis_sum[i] = dis_sum[i - 1] + dis[i];
	for(ll i = 1; i <= m; i++) {
		rd(q[i].l), rd(q[q[i].id = i].r);
		f_ans[i] = (q[i].r - q[i].l) * (dis_sum[q[i].r] - dis_sum[q[i].l - 1]);
	}
	sort(q + 1, q + 1 + m, [](const query a, const query b) {
		return bl[a.l] ^ bl[b.l]? a.l < b.l : (bl[a.l] & 1? a.r < b.r : a.r > b.r);
	});
	for(ll i = 1, l = 1, r = 0; i <= m; i++) {
		if(r < q[i].r) {
			vec1[q[i].r].pb(mkp(1, q[i].id));
			vec1[r].pb(mkp(-1, q[i].id));
			vec[l - 1].pb((Data) { r + 1, q[i].r, -1, q[i].id });
			r = q[i].r;
		}
		if(l > q[i].l) {
			vec[r].pb((Data) { q[i].l, l - 1, 1, q[i].id });
			vec2[l - 1].pb(mkp(-1, q[i].id));
			vec2[q[i].l - 1].pb(mkp(1, q[i].id));
			l = q[i].l;
		}
		if(r > q[i].r) {
			vec1[r].pb(mkp(-1, q[i].id));
			vec1[q[i].r].pb(mkp(1, q[i].id));
			vec[l - 1].pb((Data) { q[i].r + 1, r, 1, q[i].id });
			r = q[i].r;
		}
		if(l < q[i].l) {
			vec[r].pb((Data) { l, q[i].l - 1, -1, q[i].id });
			vec2[q[i].l - 1].pb(mkp(1, q[i].id));
			vec2[l - 1].pb(mkp(-1, q[i].id));
			l = q[i].l;
		}
	}
	for(ll i = 1, sum1 = 0, sum2 = 0; i <= n; i++) {
		ll tmp = ask(i); sum1 += tmp, sum2 += dis[i] + tmp;
		for(pir t: vec1[i])
			ans[t.se] += sum1 * t.fi;
		for(pir t: vec2[i])
			ans[t.se] += sum2 * t.fi;
		upd(i);
		for(Data t: vec[i]) {
			ull ret = 0;
			for(ll j = t.l; j <= t.r; j++) ret += ask(j);
			ans[t.id] += ret * t.w;
		}
	}
	for(ll i = 1; i <= m; i++) ans[q[i].id] += ans[q[i - 1].id];
	for(ll i = 1; i <= m; i++) printf("%u\n", f_ans[i] - 2 * ans[i]);
	return 0;
}

标签:ft,分块,Top,Fa,Cluster,maxn,树簇,ll,界点
From: https://www.cnblogs.com/Sktn0089/p/18595045

相关文章

  • Redis Cluster集群模式部署X
    RedisCluster模式部署Redis的哨兵模式基本已经可以实现高可用,读写分离,但是在这种模式下每台Redis服务器都存储相同的数据,很浪费内存,所以在redis3.0上加入了Cluster集群模式,实现了Redis的分布式存储,也就是说每台Redis节点上存储不同的内容。下面是Cluster集群模式的一......
  • 骨传导耳机哪个牌子好?热卖榜单TOP5机型实力解析测评
    在快节奏的现代生活中,音乐不仅成为了人们日常通勤、健身锻炼以及休闲娱乐时不可或缺的精神伴侣,而且随着科技的进步,骨传导耳机以其独特的声音传导方式和开放式的佩戴体验,逐渐成为追求高品质生活人群的新选择。这种新型耳机通过振动颅骨直接传递声音至内耳,既解放了耳朵又提升了安......
  • QGRL: Quaternion Graph Representation Learning for Heterogeneous Feature Data Cl
    QGRL:QuaternionGraphRepresentationLearningforHeterogeneousFeatureDataClustering四元数图表示学习在异构特征数据聚类中的应用JunyangChenKDD2024广东工业大学通信作者张逸群在谱聚类方法中引入四元数,四元数是一种扩展的复数系统,可以表示为......
  • .wstop扩展名勒索数据库恢复---惜分飞
    联系:手机/微信(+8617813235971)QQ(107644445)标题:.wstop扩展名勒索数据库恢复作者:惜分飞©版权所有[未经本人同意,不得以任何形式转载,否则有进一步追究法律责任的权利.]操作系统文件被加密成.[[gmtaP2R5]].[[dataserver@airmail.cc]].wstop扩展名,类似运行的oracle数据库文......
  • 【Leetcode Top 100】94. 二叉树的中序遍历
    问题背景给定一个二叉树的根节点rootrootroot,返回它的中序遍历。数据约......
  • 文献阅读笔记|将H&E图像转换为虚拟免疫组化图像的病理学工具|Accelerating histopatho
    论文链接:https://doi.org/10.1038/s42256-024-00889-5论文信息:发表于NatureMachineIntelligence。2023年12月4日投稿,2024年7月29日接收,2024年9月9日online目录AbstractIntroduction1、从HE染色病理图像合成多重免疫组化(IHC)染色图像的意义2、虚拟染色【1】含义介绍【2】配对模......
  • TOPSIS法
    TOPSIS法:多属性决策的有效工具在多属性决策分析领域,TOPSIS法(TechniqueforOrderPreferencebySimilaritytoIdealSolution)是一种广泛应用且极具价值的方法。它为解决复杂的决策问题提供了一种系统、科学的途径,尤其在面临多个评价对象和多个评价指标时,能够帮助决策者......
  • Luogu EI 的第六分块 // KTT 学习记录
    P5693EI的第六分块题目描述给定一个整数序列,支持区间加正整数以及查询区间最大子段和。思路使用线段树记录四个信息来维护答案:\(sum_i\):区间和;\(lmax_i\):最大前缀和;\(rmax_i\):最大后缀和;\(mx_i\):最大子段和。合并时我们分类讨论:\(lmax=\max(lmax_{ls},sum_{ls}+l......
  • 如何分块长文件中的 C 代码
    如何分块长文件中的C代码:思路与实现当C文件代码过长,超出大模型的输入限制时,可以通过合理的代码分块方法,将代码拆分成逻辑片段供模型逐一处理。以下是几个可行的分块方法以及具体实现思路。分块方法按功能模块分块核心思路:将文件拆分成以函数、结构体或宏定义为单......
  • 【Leetcode Top 100】146. LRU 缓存
    问题背景请你设计并实现一个满足LRU(最近最少使用)缓存约束的数据结构。实现LRUCache类:LRUCache(intcapacity)以正整数作为容量cap......