1 条题解
-
0
最优策略解释
1. 理解问题背景
我们有一个由\(N\)个农场组成的网络,农场之间通过\(N - 1\)条道路连接,形成了一棵树状结构(任意两个农场之间有且仅有一条路径)。最初只有农场\(1\)有一头病牛,疾病传播有两种方式:在农场内病牛数量翻倍,或者病牛沿道路移动到相邻农场。
2. 分析最优策略的组成部分
病牛移动策略
- 对于树状结构,要让所有农场都被感染,病牛最终必须从农场\(1\)(根节点)沿着道路移动到每个叶子节点(没有子节点的农场)。因为树中从根到叶子的路径是唯一的,所以至少需要\(N - 1\)次移动才能确保每个农场都有机会被感染(因为除了根节点,其他\(N - 1\)个节点都需要通过移动来获得病牛)。这就是代码中最初将
ans
初始化为\(N - 1\)的原因。
病牛数量翻倍策略
- 考虑非叶子节点(度数大于\(1\)的节点或根节点)。假设一个非叶子节点有\(k\)个子节点(对于根节点,所有相连节点都是子节点;对于非根非叶子节点,子节点数量比度数少\(1\),因为有一条边连接到父节点)。
- 为了最快地将病牛传播到这\(k\)个子节点,我们希望在该节点积累足够多的病牛然后再向子节点扩散。积累病牛的最快方式就是利用翻倍操作。
- 例如,如果有\(3\)个子节点,我们至少需要\(2\)次翻倍操作(最初\(1\)头牛,翻倍后\(2\)头牛,再翻倍后\(4\)头牛,这样就可以向\(3\)个子节点各派出\(1\)头牛)。一般来说,要向\(k\)个子节点传播病牛,我们需要计算满足\(2^n\geq k + 1\)的最小\(n\),这个\(n\)就是在该节点需要的翻倍操作次数。这就是代码中计算
log_children
的部分。通过这种方式,我们可以在最少的天数内让病牛从该节点传播到所有子节点。
3. 整体最优性
按照上述移动和翻倍策略,我们可以保证以最少的天数让疾病传播到整个农场网络。先通过合理的翻倍操作在各个节点积累足够的病牛,然后通过移动将病牛送到每个农场,这样的组合方式可以使传播过程最快完成,因为我们在每个节点都以最快的方式进行了病牛数量的扩充和传播。所以这种计算方式得到的结果是最优的最少天数。
// 考察数据结构:树 // 解题点:N个节点 N-1 条边,说明该图无环,可以当树处理 #include <iostream> #include <vector> #include <algorithm> using namespace std; #define MAXN 100000 int N; int d[MAXN]; int main() { // 读取农场数量N cin >> N; int a, b; // 读取N - 1条道路的信息,构建树的结构表示农场之间的连接关系 // 同时计算每个农场的度数(连接的道路数量) for (int i = 0; i < N - 1; i++) { cin >> a >> b; a--, b--; // 将农场编号调整为从0开始(方便后续数组操作) d[a]++, d[b]++; } int ans = N - 1; // 初始化答案为移动事件的数量(最坏情况,每个农场都需要通过移动感染) // 遍历每个农场 for (int i = 0; i < N; i++) { // 如果农场不是叶子节点(度数大于1)或者是根节点(编号为0) if (d[i] > 1 || i == 0) { int children = d[i]; if (i!= 0) children--; // 如果不是根节点,要减去连接到父节点的那条边(因为在计算翻倍次数时不需要考虑父节点方向) // 计算以当前节点为根的子树中,将感染传播到所有子节点所需的翻倍事件次数 // 这里通过不断计算2的幂次方来找到大于等于子节点数量 + 1的最小幂次方,即计算ceil(log(children + 1)) int log_children = 0; int pow = 1; while (pow < children + 1) log_children++, pow *= 2; ans += log_children; } } // 输出结果 cout << ans << '\n'; }
- 对于树状结构,要让所有农场都被感染,病牛最终必须从农场\(1\)(根节点)沿着道路移动到每个叶子节点(没有子节点的农场)。因为树中从根到叶子的路径是唯一的,所以至少需要\(N - 1\)次移动才能确保每个农场都有机会被感染(因为除了根节点,其他\(N - 1\)个节点都需要通过移动来获得病牛)。这就是代码中最初将
信息
- ID
- 10
- 时间
- 1000ms
- 内存
- 256MiB
- 难度
- 10
- 标签
- 递交数
- 1
- 已通过
- 1
- 上传者