CF2018C Tree Pruning
分析
好像官方题解是反向求解的,这里提供一个正向求解的思路,即直接求出最后所有叶节点到根的距离相同为 \(x\) 时需要删除的结点数 \(ans_x\) 。
如果我们最后到根的相同距离为 \(x\),那么答案有两个组成部分。
第一个部分,若到根距离为 \(x\) 的结点是一个中间结点,也就是说这个结点还有子节点,那么直接把比这个点更深的子树都删掉即可。
第二个部分,对于每一个分枝,若这个分支的最深深度小于 \(x\) ,那么这一整个分支都得删掉。因为这个分枝只要有剩余,它的叶节点到根的距离一定小于 \(x\) 。只有被完全剪掉,才会不剩余叶节点,才会没有到根距离小于 \(x\) 的叶节点。
因此,我们只需要处理出所有剪完后叶节点到根的距离的删减数,再所有答案取最小就行了,而这个过程,也只需要一次 DFS 就能完成。
对于第一个部分,只需要在 DFS 的过程维护一下下方子树的大小,在每一层加上所有下方子树大小即可。
对于第二个部分,对于每一个点,我们记录一下这个点所代表的子树的最深深度,那么想一下,是不是只要最后要保留的叶节点到根的距离大于这个结点的深度,这个结点就得被删掉。
所以,我们可以用 差分数组 来维护这一件事。
记 \(mx_i\) 为结点 \(i\) 的子树的最深深度,\(z\) 数组是第二部分的答案数组的差分数组,因此每一个点对第二部分的答案数组贡献就是对 \(z_{mx_i + 1} + 1\) 。
最后,把两部分的答案加一下就行了,然后对所有情况的答案取最小即可。
上代码!
AC CODE
#include <bits/stdc++.h>
#define int long long
#define inf INT64_MAX
#define ull unsigned long long
using namespace std;
const int N = 6e5 + 9;
int d[N], k[N], sz[N], z[N];
//d:当前结点深度 k:保留的距离要减掉的结点数 sz:当前结点代表的子树大小 z:第二部分答案的差分数组
int mx[N];//当前结点最深深度
int dmx;
vector<int> g[N];
void dfs(int st, int pre)//st:当前结点 pre:父节点
{
if(st != 1)d[st] = d[pre] + 1;
dmx = max(dmx, d[st]);//求出整棵树的最深深度
sz[st] = 1;
if(g[st].size() == 1 && st != 1)
{
mx[st] = d[st];
}
for(auto &i : g[st])
{
if(i == pre)continue;
dfs(i, st);
k[d[st]] += sz[i];//第一部分答案
sz[st] += sz[i];
mx[st] = max(mx[st], mx[i]);
}
if(st != 1)z[mx[st] + 1] ++;//给第二部分答案做贡献
}
void init(int n)
{
for(int i = 1;i <= n + 10;i ++)
{
g[i].clear();
k[i] = 0;
sz[i] = 0;
z[i] = 0;
mx[i] = 0;
}
dmx = 0;
}
void solve()
{
int n;cin >> n;
init(n);
for(int i = 1;i < n;i ++)
{
int u, v;cin >> u >> v;
g[u].push_back(v);
g[v].push_back(u);
}
d[1] = 0;
dfs(1, -1);
for(int i = 1;i <= dmx;i ++)z[i] += z[i - 1];//差分数组前缀和求出第二部分的答案
for(int i = 1;i <= dmx;i ++)k[i] += z[i];//两部分答案求和
int ans = inf;
for(int i = 1;i <= dmx;i ++)ans = min(ans, k[i]);
cout << ans << '\n';
}
signed main()
{
ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
int t = 1;cin >> t;
while(t --)solve();
return 0;
}
CF2018C Tree Pruning的更多相关文章
- [Swift]LeetCode814. 二叉树剪枝 | Binary Tree Pruning
We are given the head node root of a binary tree, where additionally every node's value is either a ...
- 814-Binary Tree Pruning
Description: We are given the head node root of a binary tree, where additionally every node’s value ...
- [LeetCode] Binary Tree Pruning 二叉树修剪
We are given the head node root of a binary tree, where additionally every node's value is either a ...
- Leetcode 814. Binary Tree Pruning
dfs 要点是这一句: return node.val==1 or node.left or node.right 完整代码: # Definition for a binary tree node. ...
- 814. Binary Tree Pruning(leetcode) (tree traverse)
https://leetcode.com/contest/weekly-contest-79/problems/binary-tree-pruning/ -- 814 from leetcode tr ...
- leetcode814 Binary Tree Pruning
""" We are given the head node root of a binary tree, where additionally every node's ...
- 【LeetCode】814. Binary Tree Pruning 解题报告(Python & C++)
作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 后序遍历 日期 题目地址:https://leetc ...
- LeetCode题解之Binary Tree Pruning
1.题目描述 2.问题分析 使用递归 3.代码 TreeNode* pruneTree(TreeNode* root) { if (root == NULL) return NULL; prun(ro ...
- [Leetcode] Binary Tree Pruning
題目是說,如果左右子樹都不存在又自已為0,就去掉那個子樹(設為null) recursive後序,左子樹,右子樹,然後是根 自已同時又是別人的子樹,所以要告訢根自已是不是存在 從a開始,左右子樹都不存 ...
- A Complete Tutorial on Tree Based Modeling from Scratch (in R & Python)
A Complete Tutorial on Tree Based Modeling from Scratch (in R & Python) MACHINE LEARNING PYTHON ...
随机推荐
- 中电金信:金Gien乐道 | 6月热门新闻盘点 回顾这一月的焦点事件
- StreamJsonRpc.ConnectionLostException 在请求完成之前, 与远程方的 JSON-RPC 连接已丢失
今天电脑重启之后,发现 visual studio 2022 的智能提示与报错经常性不好用,不光不能在正常时候提示代码错误信息,甚至在编译过后也不提示错误.反复重启,刚开始正常,隔一会儿就会提示什么什 ...
- FM的正交解调法
1.FM的模拟调制过程 FM信号是一种频率调制信号,其携带的信息保存在其信号的频率中,通过改变载波的频率来实现基带数据的传输. 其函数表达式如下: \[s(t) = A*cos(w_c*t + K ...
- JVM简介—1.Java内存区域
大纲 1.运行时数据区的介绍 2.运行时数据区各区域的作用 3.各个版本内存区域的变化 4.直接内存的使用和作用 5.站在线程的角度看Java内存区域 6.深入分析堆和栈的区别 7.方法的出入栈和栈上 ...
- StreamUtils
package com.redis.utils; import com.SpringUtils; import com.StringUtils; import lombok.extern.slf4j. ...
- 使用Apache commons-pool2实现高效的FTPClient连接池的方法
一. 连接池概述 频繁的建立和关闭连接,会极大的降低系统的性能,而连接池会在初始化的时候会创建一定数量的连接,每次访问只需从连接池里获取连接,使用完毕后再放回连接池,并不是直接关闭连接,这样可以保证 ...
- Windows下如何在当前目录下,打开cmd命令窗口
方法一: 在当前目录下,按下shift + 鼠标右键,会出现"在此处打开命令窗口"的字样,然后点击即可. 方法二: 在该文件夹上,按下shift + 鼠标右键,会出现"在 ...
- Qt编写的项目作品6-可视化大屏电子看板系统
一.功能特点 采用分层设计,整体总共分三级界面,一级界面是整体布局,二级界面是单个功能模块,三级界面是单个控件. 子控件包括饼图.圆环图.曲线图.柱状图.柱状分组图.横向柱状图.横向柱状分组图.合格率 ...
- 解决:The content of element type "web-app" must match "(icon?display
错误描述: The content of element type "web-app" must match"(icon?,display-name?,descripti ...
- Python 并发编程实战:优雅地使用 concurrent.futures
在 Python 多线程编程中,concurrent.futures 模块提供了一个高层的接口来异步执行可调用对象.今天,我们将通过一个循序渐进的案例,深入了解如何使用这个强大的工具. 从一个模拟场景 ...