1 条题解
-
0
自动搬运
来自洛谷,原作者为

泥土笨笨
《算法竞赛实战笔记》作者搬运于
2025-08-24 22:17:56,当前版本为作者最后更新于2020-03-01 23:25:06,作者可能在搬运后再次修改,您可在原文处查看最新版自动搬运只会搬运当前题目点赞数最高的题解,您可前往洛谷题解查看更多
以下是正文
因为题目求起点有多少个,那么我们依次枚举每个点作为起点,把它当做树根,从它出发往孩子走,看看能不能做到。
每当Farmer John沿着一条线在树上走的时候,遇到的每个点时间都往前加1。考虑从树上某个结点u走到它的孩子v,那么u和v上的时间都往前加1了。也就是说要改就父子一起改,父子之间的差是不变的。既然题目中是验证是否能完成,我们不妨贪心地看,每次不停地从父子之间来回走,直到把孩子改成12,这时候根据父子差固定,我们就能知道u现在是多少。然后再去搞下一个孩子,再确定u的值。直到把所有孩子都改好,此时u固定到了某个数。回到上一层,再由父亲把u改成12
那么这么一轮下来,最后除了根,其他点都是12了。那么根如果正好是12,说明是可以的。如果根不是12呢?其实根是1也行,因为这样就最后的时候从根的最后一个孩子停下,不走回来。这时候根和最后一个孩子的时间会差一个。除了这两种情况,其他都不行。
这样枚举每个点做根,每次根确定以后,一遍dfs,总的时间复杂度是,不超时。实际写的时候,用一个c数组表示每个点现在的时间,然后用0表示12,这样每次对12取模就行了,写起来比较方便。最终目标也是把所有点调成0.
代码如下:
#include <iostream> #include <cstdio> #include <vector> #include <cstring> using namespace std; const int MAXN = 2505; int n, c[MAXN], t[MAXN], ans; vector<int> adj[MAXN]; void dfs(int u, int f) { for (int i = 0; i < adj[u].size(); ++i) { int v = adj[u][i]; if (v == f) continue; //先递归孩子,把孩子调到0 dfs(v, u); //当前点和孩子的差不变,求出当前点的值 t[u] = (t[u] - t[v] + 12) % 12; } } int main() { // freopen("clocktree.in","r",stdin); // freopen("clocktree.out","w",stdout); scanf("%d", &n); for (int i = 1; i <= n; ++i) { scanf("%d", &c[i]); c[i] %= 12; } for (int i = 0; i < n - 1; ++i) { int u, v; scanf("%d%d", &u, &v); adj[u].push_back(v); adj[v].push_back(u); } for (int i = 1; i <= n; ++i) { //先把c数组拷贝到t数组里面,去操作t数组。免得c被改掉了下次循环数不对了。 memcpy(t, c, sizeof(t)); dfs(i, 0); //最后根上剩下0或者1就是可以的 if (t[i] == 0 || t[i] == 1) ans++; } cout << ans << endl; return 0; }后记,后来比赛完才想起来有个O(n)的二分染色算法,其他大佬写过题解了,我就不写了。
- 1
信息
- ID
- 5187
- 时间
- 2000ms
- 内存
- 250MiB
- 难度
- 4
- 标签
- 递交数
- 0
- 已通过
- 0
- 上传者