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\) ...
随机推荐
- 【ACM程序设计】最小生成树 Prim算法
最小生成树 ● 最小生成树的定义是给定一个无向图,如果它任意两个顶点都联通并且是一棵树,那么我们就称之为生成树(Spanning Tree).如果是带权值的无向图,那么权值之和最小的生成树,我们就称之 ...
- Pytorch Linear ()简单推导
pytorch,nn.Linear 下图中的A是权重矩阵,b是偏置. in_features输入样本的张量大小 out_features输出样本的张量大小 bias是偏置 # 测试代码 # batch ...
- centos 7.0 下安装FFmpeg软件 过程
这几天由于需要编写一个语音识别功能,用到了百度语音识别接口,从web端或小程序端传上来的音频文件是aac或者mp3或者wav格式的,需要使用FFmpeg进行格式转换,以符合百度api的要求. 安装FF ...
- php 迭代器的学习
在PHP中有一些预定义的类,比如迭代器类,有SPL提供.常用的几个类: Iterator------最基本的迭代器 IteratorAggregate --------可以提供一个迭代器的对象,但它本 ...
- Linux-ssh-key验证
ssh登录验证方式介绍 ssh服务登录的常用验证方式 用户/口令 基于密钥 基于用户和口令登录验证 客户端发起ssh请求,服务器会把自己的公钥发送给用户 用户会根据服务器发来的公钥对密码进行加密 加密 ...
- [题解][P5206][WC2019] 数树 (op = 1)
简要题意 给定 \(n, y\). 一张图有 \(|V| = n\) 个点,现在给出两棵树 \(T_1=G(V, E_1)\) 和 \(T_2=G(V, E_2)\). 定义这两棵树的权值 \(F(E ...
- [数学基础] 4 欧几里得算法&扩展欧几里得算法
欧几里得算法 欧几里得算法基于的性质: 若\(d|a, a|b\),则\(d|(ax+by)\) \((a,b)=(b,a~mod~b)\) 第二条性质证明: \(\because a~mod~b=a ...
- 好客租房12-JSX的注意点
1.4注意点 1React元素的属性名使用驼峰式命名法 2特殊属性名 class-className for->htmlFor 3没有子节点可以用单标签表示 4使用小括号包裹jsx const ...
- 《C Primer Plus》第六版笔记--7~10章
目录 第七章 C控制语句:分支和跳转 第八章 字符输入/输出和输入验证 第九章 函数 第十章 数组和指针 第七章 C控制语句:分支和跳转 if else 用法 if (expression) //ex ...
- DS18B20数字温度计 (一) 电气特性, 供电和接线方式
目录 DS18B20数字温度计 (一) 电气特性, 供电和接线方式 DS18B20数字温度计 (二) 测温, ROM和CRC校验 DS18B20数字温度计 (三) 1-WIRE总线ROM搜索算法 DS ...