简介

dsu on treedsu没有关系,但是dsu on tree借鉴了dsu的启发式合并的思想。

它是用来解决一类树上的询问问题,一般这种问题有以下特征:

\(1.\)只有对子树的查询;

\(2.\)没有修改。

如果满足以上特征,那么dsu on tree很可能就可以派上用场了。

算法

我们以CF600E Lomsat gelral为例。

Descrption

一棵以\(1\)为根的树有 \(n\) 个结点,每个结点的颜色是\(c_i\),每个颜色有一个编号,求树中每个子树的最多的颜色编号的和。

数据范围\(1\le n\le 10^5, 1\le c_i\le 10^5\)。

Solution

我们考虑暴力怎么写:遍历每一个节点 \(u\) ,然后把子树内的所有颜色暴力统计出来更新答案。

然后消除节点 \(u\) 的贡献,继续递归计算其他点的贡献。

复杂度\(O(n^2)\),显然是很不优美的。

然后dsu on tree就登场了。它也是一个暴力,但是它结合了轻重链剖分,将复杂度降到了\(O(nlogn)\)。

我们先跑一个\(dfs1\),记录节点\(u\)的重儿子\(heavy[u]\)。

接下来:

  • 遍历每一个节点。
  • 递归所有的轻儿子,递归进入时暴力加贡献,递归结束时暴力消除贡献。
  • 递归重儿子,递归进入时暴力加贡献,递归结束时不消除贡献。
  • 统计所有轻儿子的答案,并更新该节点的答案。
  • 删除所有轻儿子对答案的影响。

复杂度\(O(nlogn)\),可以通过本题。

Code

// Author: wlzhouzhuan
#pragma GCC optimize(2)
#pragma GCC optimize(3)
#include <bits/stdc++.h>
using namespace std; #define ll long long
#define ull unsigned long long
#define rint register int
#define rep(i, l, r) for (rint i = l; i <= r; i++)
#define per(i, l, r) for (rint i = l; i >= r; i--)
#define mset(s, _) memset(s, _, sizeof(s))
#define pb push_back
#define pii pair <int, int>
#define mp(a, b) make_pair(a, b) inline int read() {
int x = 0, neg = 1; char op = getchar();
while (!isdigit(op)) { if (op == '-') neg = -1; op = getchar(); }
while (isdigit(op)) { x = 10 * x + op - '0'; op = getchar(); }
return neg * x;
}
inline void print(int x) {
if (x < 0) { putchar('-'); x = -x; }
if (x >= 10) print(x / 10);
putchar(x % 10 + '0');
} const int N = 100005;
int n;
vector <int> adj[N];
void add(int u, int v) { adj[u].pb(v); } int sz[N], fa[N], heavy[N]; // heavy[u] 表示u的重儿子
void dfs1(int u, int f) {
sz[u] = 1, fa[u] = f;
for (auto v: adj[u]) {
if (v == f) continue;
dfs1(v, u);
sz[u] += sz[v];
if (sz[v] > sz[heavy[u]]) {
heavy[u] = v; // 更新重儿子
}
}
} long long ans[N], sum;
int cnt[N], col[N], Son, Max;
void change(int u, int val) {
cnt[col[u]] += val;
if (cnt[col[u]] > Max) Max = cnt[col[u]], sum = col[u];
else if (cnt[col[u]] == Max) sum += col[u];
for (auto v: adj[u]) {
// 由于重儿子的信息没有被删去,所以已经统计过了,不能再计算
if (v == fa[u] || v == Son) continue;
change(v, val);
}
}
void dfs2(int u, int keep) {
for (auto v: adj[u]) {
if (v == fa[u] || v == heavy[u]) continue;
dfs2(v, 0); // 遍历u的轻儿子
}
if (heavy[u]) dfs2(heavy[u], 1), Son = heavy[u];
change(u, 1), ans[u] = sum, Son = 0;
if (!keep) change(u, -1), sum = Max = 0;
} int main() {
n = read();
for (rint i = 1; i <= n; i++) {
col[i] = read();
}
for (rint i = 1; i < n; i++) {
int x = read(), y = read();
add(x, y), add(y, x);
}
int root = 1; // 题目默认1为根
dfs1(root, 0);
dfs2(root, 0);
for (rint i = 1; i <= n; i++) {
printf("%lld ", ans[i]);
}
return 0;
}

[算法学习] dsu on tree的更多相关文章

  1. 【算法】dsu on tree初探

    dsu on tree的本质是树上的启发式合并,它利用启发式合并的思想,可以将O(N^2)的暴力优化成O(NlogN),用于不带修改的子树信息查询. 具体如何实现呢?对于一个节点,继承它重儿子的信息, ...

  2. Note -「Dsu On Tree」学习笔记

    前置芝士 树连剖分及其思想,以及优化时间复杂度的原理. 讲个笑话这个东西其实和 Dsu(并查集)没什么关系. 算法本身 Dsu On Tree,一下简称 DOT,常用于解决子树间的信息合并问题. 其实 ...

  3. dsu on tree详解

    这个算法还是挺人性化的,没有什么难度 就是可能看起来有点晕什么的. 大体 思想是 利用重链刨分来优化子树内部的查询. 考虑一个问题要对每个子树都要询问一次.我们暴力显然是\(n^2\)的. 考虑一下优 ...

  4. dsu on tree入门

    先瞎扯几句 说起来我跟这个算法好像还有很深的渊源呢qwq.当时在学业水平考试的考场上,题目都做完了不会做,于是开始xjb出题.突然我想到这么一个题 看起来好像很可做的样子,然而直到考试完我都只想出来一 ...

  5. 【CodeForces】741 D. Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths(dsu on tree)

    [题意]给定n个点的树,每条边有一个小写字母a~v,求每棵子树内的最长回文路径,回文路径定义为路径上所有字母存在一种排列为回文串.n<=5*10^5. [算法]dsu on tree [题解]这 ...

  6. 【CodeForces】600 E. Lomsat gelral (dsu on tree)

    [题目]E. Lomsat gelral [题意]给定n个点的树,1为根,每个点有一种颜色ci,一种颜色占领一棵子树当且仅当子树内没有颜色的出现次数超过它,求n个答案——每棵子树的占领颜色的编号和Σc ...

  7. dsu on tree (树上启发式合并) 详解

    一直都没出过算法详解,昨天心血来潮想写一篇,于是 dsu on tree 它来了 1.前置技能 1.链式前向星(vector 建图) 2.dfs 建树 3.剖分轻重链,轻重儿子 重儿子 一个结点的所有 ...

  8. [dsu on tree]【学习笔记】

    十几天前看到zyf2000发过关于这个的题目的Blog, 今天终于去学习了一下 Codeforces原文链接 dsu on tree 简介 我也不清楚dsu是什么的英文缩写... 就像是树上的启发式合 ...

  9. [学习笔记]Dsu On Tree

    [dsu on tree][学习笔记] - Candy? - 博客园 题单: 也称:树上启发式合并 可以解决绝大部分不带修改的离线询问的子树查询问题 流程: 1.重链剖分找重儿子 2.sol:全局用桶 ...

随机推荐

  1. iview table表中使用render函数props传值出现问题

    使用iview中的table表格时避免不了使用render函数渲染自定义内容,或者渲染组件.但是在正常使用时出现了props传值无法识别, 按照官网介绍使用props如下: render: (h, p ...

  2. PAT B1061判断题

    题目描述: 判断题的评判很简单,本题就要求你写个简单的程序帮助老师判题并统计学生们判断题的得分. 输入格式: 输入在第一行给出两个不超过 100 的正整数 N 和 M,分别是学生人数和判断题数量.第二 ...

  3. 【Android Studio】Gradle统一管理版本号引用配置

    1.在根目录下的build.gradle文件下添加 ext{ .... } 中的内容 ...... // Define versions in a single place ext { // SDK ...

  4. ubantu系统之快捷键使用

    1. 文件管理器中,目录切换为可以编辑的状态: ctrl + l 2. gedit 搜索 : ctrl + h

  5. Python程序的流程

    1 """ 2 python程序的流程 3 """ 4 # ------------- 分支结构---------------- 5 # i ...

  6. 实际业务处理 Kafka 消息丢失、重复消费和顺序消费的问题

    关于 Kafka 消息丢失.重复消费和顺序消费的问题 消息丢失,消息重复消费,消息顺序消费等问题是我们使用 MQ 时不得不考虑的一个问题,下面我结合实际的业务来和你分享一下解决方案. 消息丢失问题 比 ...

  7. 【面试普通人VS高手系列】Fail-safe机制与Fail-fast机制分别有什么作用

    前段时间一个小伙伴去面试,遇到这样一个问题. "Fail-safe机制与Fail-fast机制分别有什么作用" 他说他听到这个问题的时候,脑子里满脸问号.那么今天我们来看一下,关于 ...

  8. eslint配置介绍-如何在uniapp中配置eslint

    eslint uniapp-eslint及vue-eslint配置 ESLint 是一个开源的 JavaScript 代码检查工具.可以让程序员在编码的过程中发现问题而不是在执行的过程中. 1. es ...

  9. 【直播回顾】OpenHarmony知识赋能第四期直播——标准系统HDF开发

    3月10日晚上19点,OpenHarmony开发者成长计划社群内,我们举办了​​知识赋能第四期直播课<OpenHarmony标准系统HDF框架介绍>​​,吸引了数千名开发者线上观看学习,并 ...

  10. 【面试普通人VS高手系列】lock和synchronized区别

    今天来分享一道阿里一面的面试题,"lock和synchronized的区别". 对于这个问题,看看普通人和高手的回答! 普通人: 嗯,lock是J.U.C包里面提供的锁,synch ...