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\) ...
随机推荐
- vmware安装或卸载时,显示无法打开注册表项
vmware卸载是出了名的臭名昭著,因为太难删干净了,删不干净又会有各种各样的问题.比如下文这个"无法打开注册表项" 这个我相信有很多人在重装vmware的时候遇到过,因此我来 ...
- Hadoop(四)C#连接Hive
Hive Hive将HiveQL(类sql语言)转为MapReduce,完成数据的查询与分析,减少了编写MapReduce的复杂度.它有以下优点: 学习成本低:熟悉sql就能使用 良好的数据分析:底层 ...
- 2.4 小白必看:零基础安装Linux系统(超级详细)
我们以新发布的 CentOS 8.1 为例,学习如何安装Linux系统. 准备工作: 1. 一台可以访问互联网的电脑 2. VMware Workstation安装包 3. CentOS8.1镜像文件 ...
- 1.17 想学好Linux,这些习惯必须养成(初学者必读)
不管是在生活还是工作中,每个人都会逐渐养成一些小习惯.坏习惯一旦形成就很难改正,所在在系统学习 Linux之前,给大家一些建议,刻意去培养一些好的习惯,对自己是很有利的. 学习Linux,要习惯使用命 ...
- os、json、sys、subprocess模块
os模块 import os 1.创建目录(文件夹) os.mkdir(r'a') # 相对路径 只能创建单级目录 os.makedirs(r'a\b') # 可以创建单级和多及目录 2.删除目录 o ...
- drools中Fact的equality modes
一.equality modes介绍 在drools中存在如下2种equality modes. 1.identity模式 identity:这是默认的情况.drools引擎使用IdentityHas ...
- 141. Linked List Cycle - LeetCode
Question 141. Linked List Cycle Solution 题目大意:给一个链表,判断是否存在循环,最好不要使用额外空间 思路:定义一个假节点fakeNext,遍历这个链表,判断 ...
- Yaml中特殊符号"| > |+ |-"的作用
"|",保留每行尾部的换行符\n. ">",删除每行尾部的换行符\n,则看似多行文本,则在程序中会将其视为一行. include_newlines: | ...
- Go微服务框架go-kratos实战05:分布式链路追踪 OpenTelemetry 使用
一.分布式链路追踪发展简介 1.1 分布式链路追踪介绍 关于分布式链路追踪的介绍,可以查看我前面的文章 微服务架构学习与思考(09):分布式链路追踪系统-dapper论文学习(https://www. ...
- LSP原则是什么
如果这篇文章能够帮到您,请给我一个免费的赞,谢谢QWQ! LSP原则并不难,但是地方就会把它说的很啰嗦,如果你对LSP还是感到疑惑,请往下看看. 先上代码: public class Bird { p ...