P6773 [NOI2020]命运
整体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方程

#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]命运的更多相关文章
- [NOI2020]命运
显然直接计数是不好计的,只能从 \(dp\) 这个角度来下手. 首先用最原始最直接的方法,直接在 \(dp\) 的过程中满足题目的要求. 既然问题给在一棵树上,那么必然和树脱不了关系,因此我们应该从树 ...
- DP 优化方法大杂烩 & 做题记录 I.
标 * 的是推荐阅读的部分 / 做的题目. 1. 动态 DP(DDP)算法简介 动态动态规划. 以 P4719 为例讲一讲 ddp: 1.1. 树剖解法 如果没有修改操作,那么可以设计出 DP 方案 ...
- 初中的一些OI琐屑 & APIO2020 & NOI2020
这篇文章会发布在我的博客上 https://www.cnblogs.com/dmoransky/(一个小习惯,把信息学竞赛的学习历程记录在个人博客中). 借这篇随笔回顾并简短总结一下我的初中OI(信息 ...
- 一个页面实例化两个ueditor编辑器,同样的出生却有不同的命运
今天遇到一个比较怪异的问题,有一项目需要在同一个页面上展现两个ueditor编辑器,在展现时并不任何问题,但当点击了“保存”按钮时就出错了,有其中一个ueditor在asp.net中无法获取编辑器的值 ...
- HDU 2571 命运
命运 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submissi ...
- HDUOJ----2571(命运)(简单动态规划)
命运 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submiss ...
- HDU 2571 命运 动态规划
命运 http://acm.hdu.edu.cn/showproblem.php?pid=2571 Problem Description 穿过幽谷意味着离大魔王lemon已经无限接近了!可谁能想到, ...
- HDU 2571 命运 (DP)
命运 Time Limit:1000MS Memory Limit:32768KB 64bit IO Format:%I64d & %I64u Submit Status Pr ...
- hdu2571 命运 动态规划Dp
转载请注明出处:http://blog.csdn.net/u012860063 题目链接:pid=2571" target="_blank">http://acm. ...
随机推荐
- 008 01 Android 零基础入门 01 Java基础语法 02 Java常量与变量 02 Java 中的关键字
008 01 Android 零基础入门 01 Java基础语法 02 Java常量与变量 02 Java 中的关键字 关键字 关键字就是一些有特殊意义的词 之前学习的程序中涉及到的关键字 Java中 ...
- matlab做gaussian高斯滤波
原文链接:https://blog.csdn.net/humanking7/article/details/46826105 核心提示 在Matlab中高斯滤波非常方便,主要涉及到下面两个函数: 函数 ...
- pycharm 解决PEP8问题,配置autopep8到菜单栏
autopep8是一个可以将Python代码自动排版为PEP8风格第三方包,使用它可以轻松地排版出格式优美整齐的代码.网络上有很多介绍如何在pycharm中配置autopep8的方案,但很多方案中还是 ...
- 深入浅出学Java-HashMap
一.概要 HashMap在JDK1.8之前的实现方式 数组+链表,但是在JDK1.8后对HashMap进行了底层优化,改为了由 数组+链表+红黑树实现,主要的目的是提高查找效率. 如下图所示: JDK ...
- 35岁老半路程序员的Python从0开始之路
9年的ERP程式开发与维护,继而转向一年的售前,再到三年半的跨行业务,近4的兜兜转转又转回来做程式了,不过与之前不同的,是这次是新的程序语言Python, 同时此次是为了教学生而学习! 从今天开始,正 ...
- volatile型变量语义讲解一 :对所有线程的可见性
volatile型变量语义讲解一 :对所有线程的可见性 一.volatile变量语义一的概念 当一个变量被定义成volatile之后,具备两个特性: 特性一:保证此变量对所有线程的可见性.这里的&qu ...
- HTML,Dreamweaver中的大小写
一般都不讲究,结果昨天在做框架网页的时候,这个target目标窗口却讲究起来大小写,开始没注意,mainframe和mainFrame居然不一样,涨姿势了! 2020.10.14
- 技术心得丨一种有效攻击BERT等模型的方法
Is BERT Really Robust? A Strong Baseline for Natural Language Attack on Text Classification and Enta ...
- 多测师_测试理轮_002(v模型和H模型)
为什么要测试?1.软件非正常运行或自身缺陷会引发问题2.代码和文档是人写的,难免会出错3.环境原因影响软件(内存不足,存储,数据库溢出等)4.软件测试活动是保证软件质量的关键之一 什么是测试?软件行业 ...
- .NetCore 异步编程 - async/await
前言: 这段时间开始用.netcore做公司项目,发现前辈搭的框架通篇运用了异步编程方式,也就是async/await方式,作为一个刚接触的小白,自然不太明白其中原理,最重要的是,这个玩意如果不明白基 ...