首先可以思考一下每次能删去的点有什么性质。

不难发现,每次能删去的点都是入度恰好为 \(1\) 的那些点(包括 \(a_i \rightarrow b_i\) 的有向边)。 换句话说,每次能删去的点既要是树上的叶子节点,并且不会被任意一条有向边 \(a_i \rightarrow b_i\) 指向。那么再来思考一下每个点能否走后离开。

因为 \(i\) 号点只能最后离开,那么我们将 \(i\) 看作这课树的根(因为每次只能走叶子)。你会发现如果 \(i\) 不能最后走当前仅当会存在一条路径(走有向边) \(x \rightarrow y\) 使得 \(y\) 是 \(x\) 子树中的点,于是我们单次判断就能做到 \(O(nm)\) 了。那么这个判定条件有没有更为简单的描述呢?其实是存在的,你会发现如果我们将所有树边从儿子指向父亲,那么 \(i\) 不能最后走当且仅当这张有向图存在着一个环。于是这样单次判断的复杂度就能做到 \(O(n + m)\) 了。但这样的复杂度还不够,我们可能需要换一种方式思考。

既然每次判断点不方便,我们能否考虑每条有向边对每个点的影响呢?事实上是可以的,不难发现对于任意一条有向边 \(x \rightarrow y\),在以 \(y\) 为根时以 \(x\) 为根的子树内所有点为根时 \(x \rightarrow y\) 就会在树上形成一个环,那么这些点都是不能最后删除的。那么我们怎么找到这些点呢?因为我们显然不可能每次都换根,可以先钦定 \(1\) 为树根。那么你会发现存在两种情况 \(y\) 为 \(x\) 的祖先时,令 \(f\) 为 \(x \rightarrow y\) 这条链上 \(y\) 的儿子(可以 \(O(\log n)\) 倍增求出,在 [USACO19JAN]Exercise Route P 中提到),那么这些点就会是整棵树除了以 \(f\) 为根的子树内的点。那么我们在根以及 \(f\) 上打标记差分即可。对于其他情况,这些点就会是以 \(x\) 为根的子树内的点,直接打标记即可。最终我们树上差分跑一边 \(dfs\) 即可。

这样就做完了吗?事实上并没有,你会发现你忽略了有向边之间的影响。那么怎样的情况会对答案有影响呢?当且仅当形成了一条跨子树的路径 \(x \rightarrow \cdots \rightarrow y\) 其中 \(y\) 为 \(x\) 子树内的点,并且仔细分析你会发现,如果出现这种情况那么整张图是不存在这样的删除序列的。那么判掉是否对答案有贡献只需判断这张图是否有解即可,直接拓扑排序每次加入度数为 \(1\) 的点,最终如果存在没有入队的点就无解。

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. #define rep(i, l, r) for (int i = l; i <= r; ++i)
  4. #define dep(i, l, r) for (int i = r; i >= l; --i)
  5. #define Next(i, u) for (int i = h[u]; i; i = e[i].next)
  6. const int N = 100000 + 5;
  7. const int M = 20 + 5;
  8. struct edge {
  9. int v, next;
  10. }e[N * 3];
  11. bool book[N];
  12. int n, m, u, v, tot, cnt, x[N], y[N], d[N], h[N], c[N], sz[N], dfn[N], dep[N], ans[N], f[N][M];
  13. queue <int> Q;
  14. int read() {
  15. char c; int x = 0, f = 1;
  16. c = getchar();
  17. while (c > '9' || c < '0') { if(c == '-') f = -1; c = getchar();}
  18. while (c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
  19. return x * f;
  20. }
  21. void add(int u, int v) {
  22. e[++tot].v = v, e[tot].next = h[u], h[u] = tot, ++d[v];
  23. }
  24. void dfs(int u, int fa){
  25. f[u][0] = fa, sz[u] = 1, dfn[u] = ++cnt, dep[u] = dep[fa] + 1;
  26. Next(i, u) {
  27. int v = e[i].v; if(v == fa) continue;
  28. dfs(v, u), sz[u] += sz[v];
  29. }
  30. }
  31. int find(int x, int y) {
  32. dep(i, 0, 20) if(dep[f[x][i]] > dep[y]) x = f[x][i];
  33. return x;
  34. }
  35. void calc(int u, int fa, int tmp){
  36. ans[u] = tmp + c[u];
  37. Next(i, u) {
  38. int v = e[i].v; if(v == fa) continue;
  39. calc(v, u, tmp + c[u]);
  40. }
  41. }
  42. int main() {
  43. n = read(), m = read();
  44. rep(i, 1, n - 1) u = read(), v = read(), add(u, v), add(v, u);
  45. dfs(1, 0);
  46. rep(j, 1, 20) rep(i, 1, n) f[i][j] = f[f[i][j - 1]][j - 1];
  47. rep(i, 1, m) {
  48. x[i] = u = read(), y[i] = v = read();
  49. if(dfn[v] >= dfn[u] && dfn[v] <= dfn[u] + sz[u] - 1) ++c[1], --c[find(v, u)];
  50. else ++c[u];
  51. }
  52. calc(1, 0, 0);
  53. rep(i, 1, m) add(x[i], y[i]);
  54. rep(i, 1, n) if(d[i] == 1) Q.push(i), book[i] = true;
  55. while(!Q.empty()) {
  56. int u = Q.front(); Q.pop();
  57. Next(i, u) {
  58. int v = e[i].v; if(book[v]) continue;
  59. --d[v]; if(d[v] == 1) Q.push(v), book[v] = 1;
  60. }
  61. }
  62. rep(i, 1, n) if(!book[i]) {
  63. rep(j, 1, n) puts("0");
  64. return 0;
  65. }
  66. rep(i, 1, n) printf(ans[i] > 0 ? "0\n" : "1\n");
  67. return 0;
  68. }

值得一提的是,判定性或定义型问题一定要去思考判定条件。另外,反向考虑每条边对答案的影响也是非常重要的。当发现自己的做法出现问题或考虑不全的时候,不要慌张,仔细分析看看能否以一种简单的方式解决这些问题。

[USACO18DEC]The Cow Gathering P的更多相关文章

  1. [USACO18DEC]The Cow Gathering

    Description: 给定一棵树,每次删去叶子,有m个限制,分别为(a,b)表示a需要比b先删,为每个点能否成为最后被删的点 Hint: \(n,m \le 10^5\) Solution: 手模 ...

  2. P5157 [USACO18DEC]The Cow Gathering

    首先考虑怎么check一个点是否能被最后一个删除. 可以这么建图,以这个点建有根树,边全部向上指,再加上剩下的有向边. 很明显,这里的一条边的定义就变成了只有删去这个点,才可以删去它指向的点. 因此, ...

  3. BZOJ1827[USACO 2010 Mar Gold 1.Great Cow Gathering]——树形DP

    题目描述 Bessie正在计划一年一度的奶牛大集会,来自全国各地的奶牛将来参加这一次集会.当然,她会选择最方便的地点来举办这次集会.每个奶牛居住在 N(1<=N<=100,000) 个农场 ...

  4. 【luoguP2986】[USACO10MAR]伟大的奶牛聚集Great Cow Gathering

    题目链接 先把\(1\)作为根求每个子树的\(size\),算出把\(1\)作为集会点的代价,不难发现把集会点移动到\(u\)的儿子\(v\)上后的代价为原代价-\(v\)的\(size\)*边权+( ...

  5. P2986 [USACO10MAR]伟大的奶牛聚集Great Cow Gat…

    题目描述 Bessie is planning the annual Great Cow Gathering for cows all across the country and, of cours ...

  6. 洛谷 P2986 [USACO10MAR]伟大的奶牛聚集Great Cow Gat…(树规)

    题目描述 Bessie is planning the annual Great Cow Gathering for cows all across the country and, of cours ...

  7. 洛谷 P2986 [USACO10MAR]Great Cow Gat…(树形dp+容斥原理)

    P2986 [USACO10MAR]伟大的奶牛聚集Great Cow Gat… 题目描述 Bessie is planning the annual Great Cow Gathering for c ...

  8. [USACO10MAR]伟大的奶牛聚集Great Cow Gat…

    题目描述 Bessie is planning the annual Great Cow Gathering for cows all across the country and, of cours ...

  9. 【题解】Luogu p2986 [USACO10MAR]伟大的奶牛聚集Great Cow Gat 树型dp

    题目描述 Bessie is planning the annual Great Cow Gathering for cows all across the country and, of cours ...

随机推荐

  1. Codeforces 872B:Maximum of Maximums of Minimums(思维)

    B. Maximum of Maximums of Minimums You are given an array a1, a2, ..., an consisting of n integers, ...

  2. git 上传项目到远程仓库

    电脑安装git客户端.注册github账号并登陆 到本地项目文件夹右键选择git bash here 输入个人信息(代码提交者) git config --global user.name " ...

  3. BUG 记录:移位运算与扩展欧几里得算法

    BUG 记录:移位运算与扩展欧几里得算法 起因 上个月就开始打算用C++写一个ECC的轮子(为什么?折磨自己呗!),奈何自己水平有点差,拖到现在才算写完底层的大数运算.在实现欧几里得算法的时候,我开始 ...

  4. Java实习生常规技术面试题每日十题Java基础(二)

    目录 1. JAVA 的反射机制的原理. 2.静态嵌套类(Static Nested Class)和内部类(Inner Class)的不同? 3.如何将String类型转化成Number类型. 4.什 ...

  5. layedit赋值方法(layedit.setContent赋值时报错-ReferenceError: layedit is not defined)

    layedit.setContent赋值时报错 - ReferenceError: layedit is not defined 报错代码: 控制台报错信息 解决方法: 步骤: 1)先赋值: 2)在渲 ...

  6. Android开发布局 案例一

    权重:就是在布局界面中所占的比例 实践案例: <?xml version="1.0" encoding="utf-8"?> <LinearLa ...

  7. 啥是Gossip协议?

    你好呀,我是歪歪. 元旦的时候我看到一个特别离谱的谣言啊,具体是什么内容我就不说了,我怕脏了大家的眼睛. 但是,我看到一个群里传的那叫一个绘声绘色,大家讨论的风生水起的,仿佛大家就在现场似的. 这事吧 ...

  8. centos6.5搭建Apache-虚拟主机

    一.配置基于域名的虚拟用户 1.创建虚拟用户的网页根目录 cd /usr/local/httpd/htdocs/ mkdir benetcom cd benetcom echo "<h ...

  9. mybatis学习笔记(三)

    使用mapper接口来实现数据操作 创建UserMapper接口,添加方法,方法名要与UserMappper.xml中id所一致并且接口名要与编写sql语句的xmL文件名同名.namespace 中的 ...

  10. Spark案例练习-打包提交

    关注公众号:分享电脑学习回复"百度云盘" 可以免费获取所有学习文档的代码(不定期更新)云盘目录说明:tools目录是安装包res 目录是每一个课件对应的代码和资源等doc 目录是一 ...