整体DP

很明显计算答案需要用容斥计算,如果暴力容斥的话,就是枚举哪些路径不符合条件,在这些路径的并集中的边都不能取,其他边任意取,设当前取了$i$条路径,那么对答案的贡献是$(-1)^i2^{n-1-Union}$

但是可以发现这个路径是自下往上的,可以考虑树形DP,设$dp[i][j][k]$表示在i的子树内,已选$k$条路径的上面那个端点的深度的最小值为$j$的方案数,特别的如果在这个子树内不选任何一条路径的话,$j$为$maxde$

但其实在容斥的时候,我们并不关心这个$k$的具体取值,只要关心其奇偶性即可,那么可以去掉这一维,在设初始值的时候将$-1$乘到值里面,在之后乘起来的时候,$-1$会帮助数值自动变号

考虑一个儿子一个儿子更新$dp[x][i]$,分情况讨论

如果$i\leqslant de[x]$

$dp[x][i]=\sum_{j=i+1}^{maxde}2dp[u][j]dp[x][i]+\sum_{j=i+1}^{de[x]}dp[u][j]dp[x][i]+\sum_{j=i}^{maxde}dp[u][i]dp[x][j]$

第一部分表示$u$这棵子树内的最浅的那个不能选的路径是小于当前$x$的深度,那么$u->x$这条边是可以任意选择;第二部分就表示$u$中最浅的点比$i$深,那么当前最浅的那个节点依然是$i$;第三部分表示$u$中最浅的点比$i$浅,那么当前最浅的点需要更新

如果$i>de[x]$

$dp[x][i]=\sum_{j=i+1}^{maxde}2dp[u][j]dp[x][i]+\sum_{j=i}^{maxde}2dp[u][i]dp[x][j]$

这也是类似的

可以发现第二维不为$0$的取值是较少的,只有在较前的点才会变多,那么就用线段树合并维护这个DP(跟[PKUWC2018]Minimax类似)

但是这个DP的方程需要分类讨论,就很难直观的进行维护修改,考虑如何用一个式子来表示这个DP方程

可以注意到第一个式子里前面两个部分是可以衔接在一起的,只不过第一部分有一个$2$的系数,但下标都是大于$de[x]$的,第二条方程也是一样,都是下标大于$de[x]$的时候$dp[u]$需要$*2$
那么在线段树合并之前,就把这段区间乘$2$即可
那么方程就是$dp[x][i]=\sum_{j=i+1}^{maxde}dp[x][i]dp[u][j]+\sum_{j=i+1}^{maxde}dp[x][j]dp[u][i]+dp[x][i]dp[u][i]$
再考虑新增路径的情况,对于一个节点可能会有多条下端点是这个节点的路径,设有$size$条,取第$i$浅的路径的话,那么比它浅的路径都不能选,比它深的路径随便选,那么在DP中需要赋值为钦定选这条路径和比它深的路径随便选的容斥系数之和
$\sum_{j=0}^{size-i}(-1)^{j+1}C_{size-i}^{j}$
$=(-1)\sum_{j=0}^{size-i}(-1)^{j}C_{size-i}^{j}$
$=(-1)(1-1)^{size-i}$
那么只有$i=size$的时候这个值不为$0$,那么只要记录最深的的那条路径更新即可
注意可以一条路径都不选,那么在$maxde$处要设为$1$

  1. #include <bits/stdc++.h>
  2. #define mod 998244353
  3. using namespace std;
  4. const int N=5*1e5+100;
  5. int n,m,w,de[N],maxde,last[N],root[N],cnt;
  6. int tot,first[N],nxt[N*2],point[N*2];
  7. struct node
  8. {
  9. int ls,rs;
  10. long long dp,tag;
  11. }sh[N*40];
  12. inline void add(long long &a,long long b){a=(a+b);((a>mod)?a-=mod:a=a);}
  13. inline void del(long long &a,long long b){a=(a-b+mod)%mod;}
  14. inline void mul(long long &a,long long b){a=(a*b)%mod;}
  15. inline bool cmp(int a,int b){return(de[a]<de[b]);}
  16. inline int read()
  17. {
  18. int f=1,x=0;char s=getchar();
  19. while(s<'0'||s>'9'){if(s=='-')f=-1;s=getchar();}
  20. while(s>='0'&&s<='9'){x=x*10+s-'0';s=getchar();}
  21. return x*f;
  22. }
  23. inline void add_edge(int x,int y)
  24. {
  25. tot++;
  26. nxt[tot]=first[x];
  27. first[x]=tot;
  28. point[tot]=y;
  29. }
  30. void dfs(int x,int fa)
  31. {
  32. for (int i=first[x];i!=-1;i=nxt[i])
  33. {
  34. int u=point[i];
  35. if (u==fa) continue;
  36. de[u]=de[x]+1;
  37. dfs(u,x);
  38. }
  39. }
  40. inline void pushup(int x)
  41. {
  42. sh[x].dp=(sh[sh[x].ls].dp+sh[sh[x].rs].dp)%mod;
  43. }
  44. inline void pushdown(int x)
  45. {
  46. if (sh[x].tag==1) return;
  47. if (sh[x].ls) mul(sh[sh[x].ls].dp,sh[x].tag),mul(sh[sh[x].ls].tag,sh[x].tag);
  48. if (sh[x].rs) mul(sh[sh[x].rs].dp,sh[x].tag),mul(sh[sh[x].rs].tag,sh[x].tag);
  49. sh[x].tag=1;
  50. }
  51. int insert(int x,int l,int r,int wh,int v)
  52. {
  53. if (!x) x=++cnt;
  54. sh[x].tag=1;
  55. if (l==r)
  56. {
  57. sh[x].dp=v;
  58. return x;
  59. }
  60. int mid=(l+r)>>1;
  61. if (wh<=mid) sh[x].ls=insert(sh[x].ls,l,mid,wh,v);
  62. else sh[x].rs=insert(sh[x].rs,mid+1,r,wh,v);
  63. pushup(x);
  64. return x;
  65. }
  66. void change(int x,int l,int r,int ll,int rr)
  67. {
  68. if (ll<=l && rr>=r)
  69. {
  70. mul(sh[x].dp,2);mul(sh[x].tag,2);
  71. return;
  72. }
  73. int mid=(l+r)>>1;
  74. pushdown(x);
  75. if (ll<=mid) change(sh[x].ls,l,mid,ll,rr);
  76. if (rr>mid) change(sh[x].rs,mid+1,r,ll,rr);
  77. pushup(x);
  78. }
  79. int merge(int a,int b,int l,int r,long long sa,long long sb,long long pa,long long pb)
  80. {
  81. if (!a)
  82. {
  83. mul(sh[b].dp,(sa-pa+mod)%mod);
  84. mul(sh[b].tag,(sa-pa+mod)%mod);
  85. return b;
  86. }
  87. if (!b)
  88. {
  89. mul(sh[a].dp,(sb-pb+mod)%mod);
  90. mul(sh[a].tag,(sb-pb+mod)%mod);
  91. return a;
  92. }
  93. if (l==r)
  94. {
  95. add(pa,sh[a].dp);add(pb,sh[b].dp);
  96. sh[a].dp=(sh[a].dp*(sb-pb+mod)%mod+sh[b].dp*(sa-pa+mod)%mod+sh[a].dp*sh[b].dp%mod)%mod;
  97. return a;
  98. }
  99. int mid=(l+r)>>1;long long ta=pa,tb=pb;
  100. pushdown(a);pushdown(b);
  101. add(ta,sh[sh[a].ls].dp);add(tb,sh[sh[b].ls].dp);
  102. sh[a].ls=merge(sh[a].ls,sh[b].ls,l,mid,sa,sb,pa,pb);
  103. sh[a].rs=merge(sh[a].rs,sh[b].rs,mid+1,r,sa,sb,ta,tb);
  104. pushup(a);
  105. return a;
  106. }
  107. void dfs1(int x,int fa)
  108. {
  109. if (last[x]) root[x]=insert(root[x],1,maxde,last[x],mod-1);
  110. root[x]=insert(root[x],1,maxde,maxde,1);
  111. for (int i=first[x];i!=-1;i=nxt[i])
  112. {
  113. int u=point[i];
  114. if (u==fa) continue;
  115. dfs1(u,x);
  116. change(root[u],1,maxde,de[x]+1,maxde);
  117. root[x]=merge(root[x],root[u],1,maxde,sh[root[x]].dp,sh[root[u]].dp,0,0);
  118. }
  119. }
  120. int main()
  121. {
  122. tot=-1;
  123. memset(first,-1,sizeof(first));
  124. memset(nxt,-1,sizeof(nxt));
  125. n=read();
  126. for (int i=1;i<n;i++)
  127. {
  128. int u=read(),v=read();
  129. add_edge(u,v);add_edge(v,u);
  130. }
  131. de[1]=1;
  132. dfs(1,1);
  133. for (int i=1;i<=n;i++) maxde=max(maxde,de[i]);
  134. maxde++;
  135. m=read();
  136. for (int i=1;i<=m;i++)
  137. {
  138. int u=read(),v=read();
  139. last[v]=max(last[v],de[u]);
  140. }
  141. dfs1(1,1);
  142. printf("%lld\n",sh[root[1]].dp);
  143. }

P6773 [NOI2020]命运的更多相关文章

  1. [NOI2020]命运

    显然直接计数是不好计的,只能从 \(dp\) 这个角度来下手. 首先用最原始最直接的方法,直接在 \(dp\) 的过程中满足题目的要求. 既然问题给在一棵树上,那么必然和树脱不了关系,因此我们应该从树 ...

  2. DP 优化方法大杂烩 & 做题记录 I.

    标 * 的是推荐阅读的部分 / 做的题目. 1. 动态 DP(DDP)算法简介 动态动态规划. 以 P4719 为例讲一讲 ddp: 1.1. 树剖解法 如果没有修改操作,那么可以设计出 DP 方案 ...

  3. 初中的一些OI琐屑 & APIO2020 & NOI2020

    这篇文章会发布在我的博客上 https://www.cnblogs.com/dmoransky/(一个小习惯,把信息学竞赛的学习历程记录在个人博客中). 借这篇随笔回顾并简短总结一下我的初中OI(信息 ...

  4. 一个页面实例化两个ueditor编辑器,同样的出生却有不同的命运

    今天遇到一个比较怪异的问题,有一项目需要在同一个页面上展现两个ueditor编辑器,在展现时并不任何问题,但当点击了“保存”按钮时就出错了,有其中一个ueditor在asp.net中无法获取编辑器的值 ...

  5. HDU 2571 命运

    命运 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submissi ...

  6. HDUOJ----2571(命运)(简单动态规划)

    命运 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submiss ...

  7. HDU 2571 命运 动态规划

    命运 http://acm.hdu.edu.cn/showproblem.php?pid=2571 Problem Description 穿过幽谷意味着离大魔王lemon已经无限接近了!可谁能想到, ...

  8. HDU 2571 命运 (DP)

    命运 Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u Submit Status Pr ...

  9. hdu2571 命运 动态规划Dp

    转载请注明出处:http://blog.csdn.net/u012860063 题目链接:pid=2571" target="_blank">http://acm. ...

随机推荐

  1. 手把手教你AspNetCore WebApi:增删改查

    前言 小明已经创建与运行了WebApi项目,了解项目结构有哪些组成,并学会了怎么发布到IIS.基础已经建好,从现在开始要真正实现待办事项的功能了. 新建表 CREATE TABLE [dbo].[To ...

  2. CString类常用方法----Left(),Mid(),Right()

    参考:https://blog.csdn.net/Qingqinglanghua/article/details/4992624 CString Left( int nCount ) const;   ...

  3. 01 . OpenResty简介部署,优缺点,压测,适用场景及用Lua实现服务灰度发布

    简介 OpenResty 是一个基于 Nginx 与 Lua 的高性能 Web 平台,其内部集成了大量精良的 Lua 库.第三方模块以及大多数的依赖项.用于方便地搭建能够处理超高并发.扩展性极高的动态 ...

  4. 达梦产品技术支持培训-day7-DM8数据库备份与还原-原理

    (本文部分内容摘自DM产品技术支持培训文档,如需要更详细的文档,请查询官方操作手册,谢谢) 1.DM8备份还原简介 1.1.基本概念 (1)表空间与数据文件 ▷ DM8表空间类型: ▷ SYSTEM ...

  5. axio跨域请求,vue中的config的配置项。

    这是我用 vue cli 脚手架搭建的跨域.以上是可以请求到的.

  6. 多测师讲解python __for 循环___高级讲师肖sir

    横向输出 1.遍历字符串 2.遍历列表 3.遍历元组 方法一: 方法二: 方法三: #循环字典:方法一# dict1={"name":"zhihao",&quo ...

  7. 爬虫之Selenium

    简介 selenium最初是一个自动化测试工具,而爬虫中使用它主要是为了解决requests无法直接执行JavaScript代码的问题 selenium本质是通过驱动浏览器,完全模拟浏览器的操作,比如 ...

  8. pyquery 匹配NavigableString

    pyquery 匹配NavigableString不像xpath那样精确找打匹配对象,只需匹配包含NavigableString的根节点

  9. Linux安装软件时90%的人会遇到这个报错,如何解决?

    提示 Could not get lock /var/lib/dpkg/lock 报错? 有些小伙伴在使用 apt 包管理器更新或安装软件时,可能会遇到过诸如以下的错误提示: E: Could not ...

  10. Apache Hudi助力nClouds加速数据交付

    1. 概述 在nClouds上,当客户的业务决策取决于对近实时数据的访问时,客户通常会向我们寻求有关数据和分析平台的解决方案.但随着每天创建和收集的数据量都在增加,这使得使用传统技术进行数据分析成为一 ...