整体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$

#include <bits/stdc++.h>
#define mod 998244353
using namespace std;
const int N=5*1e5+100;
int n,m,w,de[N],maxde,last[N],root[N],cnt;
int tot,first[N],nxt[N*2],point[N*2];
struct node
{
int ls,rs;
long long dp,tag;
}sh[N*40];
inline void add(long long &a,long long b){a=(a+b);((a>mod)?a-=mod:a=a);}
inline void del(long long &a,long long b){a=(a-b+mod)%mod;}
inline void mul(long long &a,long long b){a=(a*b)%mod;}
inline bool cmp(int a,int b){return(de[a]<de[b]);}
inline int read()
{
int f=1,x=0;char s=getchar();
while(s<'0'||s>'9'){if(s=='-')f=-1;s=getchar();}
while(s>='0'&&s<='9'){x=x*10+s-'0';s=getchar();}
return x*f;
}
inline void add_edge(int x,int y)
{
tot++;
nxt[tot]=first[x];
first[x]=tot;
point[tot]=y;
}
void dfs(int x,int fa)
{
for (int i=first[x];i!=-1;i=nxt[i])
{
int u=point[i];
if (u==fa) continue;
de[u]=de[x]+1;
dfs(u,x);
}
}
inline void pushup(int x)
{
sh[x].dp=(sh[sh[x].ls].dp+sh[sh[x].rs].dp)%mod;
}
inline void pushdown(int x)
{
if (sh[x].tag==1) return;
if (sh[x].ls) mul(sh[sh[x].ls].dp,sh[x].tag),mul(sh[sh[x].ls].tag,sh[x].tag);
if (sh[x].rs) mul(sh[sh[x].rs].dp,sh[x].tag),mul(sh[sh[x].rs].tag,sh[x].tag);
sh[x].tag=1;
}
int insert(int x,int l,int r,int wh,int v)
{
if (!x) x=++cnt;
sh[x].tag=1;
if (l==r)
{
sh[x].dp=v;
return x;
}
int mid=(l+r)>>1;
if (wh<=mid) sh[x].ls=insert(sh[x].ls,l,mid,wh,v);
else sh[x].rs=insert(sh[x].rs,mid+1,r,wh,v);
pushup(x);
return x;
}
void change(int x,int l,int r,int ll,int rr)
{
if (ll<=l && rr>=r)
{
mul(sh[x].dp,2);mul(sh[x].tag,2);
return;
}
int mid=(l+r)>>1;
pushdown(x);
if (ll<=mid) change(sh[x].ls,l,mid,ll,rr);
if (rr>mid) change(sh[x].rs,mid+1,r,ll,rr);
pushup(x);
}
int merge(int a,int b,int l,int r,long long sa,long long sb,long long pa,long long pb)
{
if (!a)
{
mul(sh[b].dp,(sa-pa+mod)%mod);
mul(sh[b].tag,(sa-pa+mod)%mod);
return b;
}
if (!b)
{
mul(sh[a].dp,(sb-pb+mod)%mod);
mul(sh[a].tag,(sb-pb+mod)%mod);
return a;
}
if (l==r)
{
add(pa,sh[a].dp);add(pb,sh[b].dp);
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;
return a;
}
int mid=(l+r)>>1;long long ta=pa,tb=pb;
pushdown(a);pushdown(b);
add(ta,sh[sh[a].ls].dp);add(tb,sh[sh[b].ls].dp);
sh[a].ls=merge(sh[a].ls,sh[b].ls,l,mid,sa,sb,pa,pb);
sh[a].rs=merge(sh[a].rs,sh[b].rs,mid+1,r,sa,sb,ta,tb);
pushup(a);
return a;
}
void dfs1(int x,int fa)
{
if (last[x]) root[x]=insert(root[x],1,maxde,last[x],mod-1);
root[x]=insert(root[x],1,maxde,maxde,1);
for (int i=first[x];i!=-1;i=nxt[i])
{
int u=point[i];
if (u==fa) continue;
dfs1(u,x);
change(root[u],1,maxde,de[x]+1,maxde);
root[x]=merge(root[x],root[u],1,maxde,sh[root[x]].dp,sh[root[u]].dp,0,0);
}
}
int main()
{
tot=-1;
memset(first,-1,sizeof(first));
memset(nxt,-1,sizeof(nxt));
n=read();
for (int i=1;i<n;i++)
{
int u=read(),v=read();
add_edge(u,v);add_edge(v,u);
}
de[1]=1;
dfs(1,1);
for (int i=1;i<=n;i++) maxde=max(maxde,de[i]);
maxde++;
m=read();
for (int i=1;i<=m;i++)
{
int u=read(),v=read();
last[v]=max(last[v],de[u]);
}
dfs1(1,1);
printf("%lld\n",sh[root[1]].dp);
}

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. 008 01 Android 零基础入门 01 Java基础语法 02 Java常量与变量 02 Java 中的关键字

    008 01 Android 零基础入门 01 Java基础语法 02 Java常量与变量 02 Java 中的关键字 关键字 关键字就是一些有特殊意义的词 之前学习的程序中涉及到的关键字 Java中 ...

  2. matlab做gaussian高斯滤波

    原文链接:https://blog.csdn.net/humanking7/article/details/46826105 核心提示 在Matlab中高斯滤波非常方便,主要涉及到下面两个函数: 函数 ...

  3. pycharm 解决PEP8问题,配置autopep8到菜单栏

    autopep8是一个可以将Python代码自动排版为PEP8风格第三方包,使用它可以轻松地排版出格式优美整齐的代码.网络上有很多介绍如何在pycharm中配置autopep8的方案,但很多方案中还是 ...

  4. 深入浅出学Java-HashMap

    一.概要 HashMap在JDK1.8之前的实现方式 数组+链表,但是在JDK1.8后对HashMap进行了底层优化,改为了由 数组+链表+红黑树实现,主要的目的是提高查找效率. 如下图所示: JDK ...

  5. 35岁老半路程序员的Python从0开始之路

    9年的ERP程式开发与维护,继而转向一年的售前,再到三年半的跨行业务,近4的兜兜转转又转回来做程式了,不过与之前不同的,是这次是新的程序语言Python, 同时此次是为了教学生而学习! 从今天开始,正 ...

  6. volatile型变量语义讲解一 :对所有线程的可见性

    volatile型变量语义讲解一 :对所有线程的可见性 一.volatile变量语义一的概念 当一个变量被定义成volatile之后,具备两个特性: 特性一:保证此变量对所有线程的可见性.这里的&qu ...

  7. HTML,Dreamweaver中的大小写

    一般都不讲究,结果昨天在做框架网页的时候,这个target目标窗口却讲究起来大小写,开始没注意,mainframe和mainFrame居然不一样,涨姿势了! 2020.10.14

  8. 技术心得丨一种有效攻击BERT等模型的方法

    Is BERT Really Robust? A Strong Baseline for Natural Language Attack on Text Classification and Enta ...

  9. 多测师_测试理轮_002(v模型和H模型)

    为什么要测试?1.软件非正常运行或自身缺陷会引发问题2.代码和文档是人写的,难免会出错3.环境原因影响软件(内存不足,存储,数据库溢出等)4.软件测试活动是保证软件质量的关键之一 什么是测试?软件行业 ...

  10. .NetCore 异步编程 - async/await

    前言: 这段时间开始用.netcore做公司项目,发现前辈搭的框架通篇运用了异步编程方式,也就是async/await方式,作为一个刚接触的小白,自然不太明白其中原理,最重要的是,这个玩意如果不明白基 ...