CF778C 题解
题目连接就不放了,人类应该反对阴间题目描述
下面给出一个作为正常人能够看懂的题面描述:
给一棵
树,可以删掉某一层的所有节点和边。
被删除的节点的子节点会代替当前节点,形成新的一层节点,如果有相同的可以合并。
求删掉哪一层边后合并出的树最小?
考虑
我们假设已经删去了一个节点,观察下一步会发生什么。
那么接下来一定是这个节点的儿子们所代表的子树合并成一个
然后我们来考虑这样一段
inline int merge(int x, int y) {
if (x == 0 || y == 0)
return x + y;
int now = ++cnt;
for (int i = 0; i < 26; i++)
son[now][i] = merge(son[x][i], son[y][i]);
return now;
}
相信大家对
下面我们来考虑新建一个节点的意义:
我们发现在进行一次新建节点时一定是把其中的两个节点合并成为一个点。
所以也就可以直接等价于是删除了原来树上的一个节点。
我们用
这是删除一个点的情况。
那么我们要删除整整一个层怎么办呢?
直接用
Code#
#include <bits/stdc++.h>
#define file(a) freopen(a".in", "r", stdin), freopen(a".out", "w", stdout)
#define Enter putchar('\n')
#define quad putchar(' ')
const int N = 600005;
int n, son[N][30], root, ans, maxn, sum[N], cnt;
char c[10];
inline int merge(int x, int y) {
if (x == 0 || y == 0)
return x + y;
int now = ++cnt;
for (int i = 0; i < 26; i++)
son[now][i] = merge(son[x][i], son[y][i]);
return now;
}
inline void dfs(int now, int deep) {
int num;
num = cnt = n + 1;
for (int i = 0; i < 26; i++)
if (son[now][i])
num = merge(num, son[now][i]);
sum[deep] += cnt - n - 1;
for (int i = 0; i < 26; i++)
if (son[now][i]) dfs(son[now][i], deep + 1);
}
signed main(void) {
// file("CF778C");
std::ios::sync_with_stdio(false);
std::cin.tie(0);
std::cout.tie(0);
std::cin >> n;
root = 1;
for (int i = 1, x, y; i < n; i++) {
std::cin >> x >> y >> c;
son[x][c[0] - 'a'] = y;
}
dfs(1, 1);
for (int i = 1; i <= n; i++)
if (sum[i] > maxn) {
maxn = sum[i];
ans = i;
}
std::cout << n - maxn << std::endl << ans << std::endl;
}
作者:Aonynation
出处:https://www.cnblogs.com/Oier-GGG/p/16211508.html
版权:本作品采用「署名-非商业性使用-相同方式共享 4.0 国际」许可协议进行许可。
分类:
题解
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
· 地球OL攻略 —— 某应届生求职总结
· 提示词工程——AI应用必不可少的技术
· Open-Sora 2.0 重磅开源!
· 周边上新:园子的第一款马克杯温暖上架