Solution -「Local 11145」诗意狗
0x01 前置芝士
树形结构?贪心?思维?眼睛?
好有趣。。。 link
0x02
题目大意:给你一颗有 \(n\) 个节点的树,你需要尽可能多的删掉边,使得剩下的图中有 \(k\) 个点满足互相能走到。求最后剩下的边数。
我们深度剖析出题人其实是想考树形 \(dp\) 的,可是呢,这其实就是道一眼题。
首先,我们剩下的图最优情况一定是只剩 \(k\) 个点。
对于答案,最优的情况其实就是 \(k\) 个点分成 \(\lfloor \frac k 2 \rfloor\) 堆,使得每一堆只有两个点或有一堆有三个点。这也是最理想的情况,但显然一些树无法满足。不过你会发现,这个理想情况最后剩的边为 \(\lceil \frac k 2 \rceil\)。
于是我们考虑原树可以拆分成多少个多少堆,使得每一堆的点数不大于 \(2\),记能拆分出的个数为 \(cnt\)。
如果 \(cnt >= \lfloor \frac k 2 \rfloor\),也就是说这棵树可以拆分出这么多堆,那么答案就是 \(\lceil \frac k 2 \rceil\)。
而如果 \(cnt < \lfloor \frac k 2 \rfloor\),也就是说这棵树是不支持最优解的,那么我们就运用一下贪心。首先我们需要剩余 \(k\) 个点,而因为我们原树只能拆出 \(cnt\) 堆,所以说这 \(k\) 个点中,一定有 \(cnt \times 2\) 个点可以成为理想情况。那么我们就考虑剩下 \(k - cnt \times 2\) 个点怎么办。其实我也不知道怎么办,但我知道这些点一定能通过一条边和已经处于理想情况的一堆连接起来,那么累加边即可。此时答案为 \(k - cnt \times 2 + cnt\),即 \(k - cnt\)。
把这两种情况综合一下就会得到答案:\(k - \min(\lfloor \frac k 2 \rfloor, cnt)\)。
关于 \(cnt\) 的求法,我们可以直接遍历这棵树。
对于节点 \(u\),它的子节点 \(v\),如果所有的 \(v\) 被取用过,我们就不取,并把 \(u\) 设为未被取用;如果有一个 \(v\) 没有被取用,我们就取没被取用的第一个 \(v\) 并把 \(u\) 设为被取用。可以结合代码分析。
0x03 code
#include <cstdio>
#include <vector>
#include <algorithm>
using namespace std;
typedef long long LL;
int read() {
int x = 0, k = 1;
char s = getchar();
while (s < '0' || s > '9') {
if (s == '-')
k = -1;
s = getchar();
}
while (s >= '0' && s <= '9') {
x = (x << 3) + (x << 1) + (s ^ 48);
s = getchar();
}
return x * k;
}
void write(int x) {
if (x < 0) {
putchar('-');
x = -x;
}
if (x > 9)
write(x / 10);
putchar(x % 10 + '0');
}
void print(int x, char s) {
write(x);
putchar(s);
}
int Min(int x, int y) { return x < y ? x : y; }
const int MAXN = 1e5 + 5;
int son[MAXN], dp[MAXN];
vector<int> mp[MAXN];
void Add_Edge(int u, int v) {
mp[u].push_back(v);
mp[v].push_back(u);
}
int cnt = 0;
bool Tree_Dp(int u, int fa) {
bool f = 1;
for (int i = 0; i < mp[u].size(); i++) {
int v = mp[u][i];
if (v == fa)
continue;
int t = Tree_Dp(v, u);
if (f && t) {
f = 0;
cnt++;
}
}
return f;
}
int main() {
int T = read();
while (T--) {
int n = read(), k = read();
for (int i = 1; i <= n; i++) mp[i].clear();
for (int i = 1; i < n; i++) {
int u = read(), v = i + 1;
Add_Edge(u, v);
}
for (int i = 1; i <= n; i++) son[i] = mp[i].size();
cnt = 0;
Tree_Dp(1, 0);
print(k - Min(cnt, k >> 1), '\n');
}
return 0;
}
Solution -「Local 11145」诗意狗的更多相关文章
- Solution -「ARC 104E」Random LIS
\(\mathcal{Description}\) Link. 给定整数序列 \(\{a_n\}\),对于整数序列 \(\{b_n\}\),\(b_i\) 在 \([1,a_i]\) 中等概率 ...
- Solution -「LOCAL」二进制的世界
\(\mathcal{Description}\) OurOJ. 给定序列 \(\{a_n\}\) 和一个二元运算 \(\operatorname{op}\in\{\operatorname{ ...
- Solution -「LOCAL」大括号树
\(\mathcal{Description}\) OurTeam & OurOJ. 给定一棵 \(n\) 个顶点的树,每个顶点标有字符 ( 或 ).将从 \(u\) 到 \(v\) ...
- Solution -「LOCAL」过河
\(\mathcal{Description}\) 一段坐标轴 \([0,L]\),从 \(0\) 出发,每次可以 \(+a\) 或 \(-b\),但不能越出 \([0,L]\).求可达的整点数. ...
- Solution -「LOCAL」Drainage System
\(\mathcal{Description}\) 合并果子,初始果子的权值在 \(1\sim n\) 之间,权值为 \(i\) 的有 \(a_i\) 个.每次可以挑 \(x\in[L,R]\) ...
- Solution -「LOCAL」Burning Flowers
灼之花好评,条条生日快乐(假装现在 8.15)! \(\mathcal{Description}\) 给定一棵以 \(1\) 为根的树,第 \(i\) 个结点有颜色 \(c_i\) 和光亮值 ...
- Solution -「LOCAL」画画图
\(\mathcal{Description}\) OurTeam. 给定一棵 \(n\) 个点的树形随机的带边权树,求所有含奇数条边的路径中位数之和.树形生成方式为随机取不连通两点连边直到全 ...
- Solution -「LOCAL」ZB 平衡树
\(\mathcal{Description}\) OurOJ. 维护一列二元组 \((a,b)\),给定初始 \(n\) 个元素,接下来 \(m\) 次操作: 在某个位置插入一个二元组: 翻 ...
- Solution -「LOCAL」舟游
\(\mathcal{Description}\) \(n\) 中卡牌,每种三张.对于一次 \(m\) 连抽,前 \(m-1\) 次抽到第 \(i\) 种的概率是 \(p_i\),第 \(m\) ...
随机推荐
- GitStats - 统计Git所有提交记录工具
如果你是研发效能组的一员或者在从事 CI/CD 或 DevOps,除了提供基础设施,指标和数据是也是一个很重要的一环,比如需要分析下某个 Git 仓库代码提交情况: 该仓库的代码谁提交的代码最多 该仓 ...
- 五三想休息,今天还学习,图解二叉树的层序遍历BFS(广度优先)模板,附面试题题解
壹 ❀ 引 我在从JS执行栈角度图解递归以及二叉树的前.中.后遍历的底层差异一文中,从一个最基本的数组遍历引出递归,在掌握递归的书写规则后,又从JS执行栈角度解释了二叉树三种深度优先(前序.中序后序) ...
- Linux入门进阶 - 如何在Linux中使用export命令
来自:Linux迷链接:https://www.linuxmi.com/linux-export.html Linux export命令会标记哪些值需要传递给一组子进程.这是bash shell提供的 ...
- kNN-画图
现在我们想要展示一些可视化内容 首先导包,如果是在jupyter notebook上,需要加入魔法函数:%matplotlib inline,这表示可以在jupyter上直接画图 import dat ...
- 我怀疑这是IDEA的BUG,但是我翻遍全网没找到证据!
你好呀,我是歪歪. 前几天有朋友给我发来这样的一个截图: 他说他不理解,为什么这样不报错. 我说我也不理解,把一个 boolean 类型赋值给 int 类型,怎么会不报错呢,并接着追问他:这个代码截图 ...
- windows 10 21H1 顶部任务栏点击音量或其他图标不出弹框
右键任务栏,按照图片中描述操作
- 解决 js aysnc await try-catch 地狱
- JavaSE_关键字 接口 代码块 枚举
1 Java中的关键字 1.1 static关键字 static特点 : 静态成员被所在类的所有对象共享 随着类的加载而加载 , 优先于对象存在 可以通过对象调用 , 也可以通过类名调用 , 建议使用 ...
- .Net CLR GC动态获取函数头地址,C++的骚操作(慎入)
前言: 太懒了,从没有在这里正儿八经的写过文章.看到一些人的高产,真是惭愧.决定稍微变得不那么懒.如有疏漏,请指正. .net的GC都谈的很多了,本篇主要是剑走偏锋,聊聊一些个人认为较为核心的细节方面 ...
- 【RocketMQ】MQ消息发送
消息发送 首先来看一个RcoketMQ发送消息的例子: @Service public class MQService { @Autowired DefaultMQProducer defaultMQ ...