题目描述

给定一棵 n 个结点的树,你从点 x 出发,每次等概率随机选择一条与所在点相邻的边走过去。

有 Q 次询问,每次询问给定一个集合 S,求如果从 x 出发一直随机游走,直到点集 S 中所有点都至少经过一次的话,期望游走几步。

特别地,点 x(即起点)视为一开始就被经过了一次。

答案对 998244353 取模。

输入格式

第一行三个正整数 n,Q,x。

接下来 n-1 行,每行两个正整数 (u,v) 描述一条树边。

接下来 Q 行,每行第一个数 k 表示集合大小,接下来 k 个互不相同的数表示集合 S。

输出格式

输出 Q 行,每行一个非负整数表示答案。

样例输入

3 5 1
1 2
2 3
1 1
1 3
2 2 3
3 1 2 3
2 1 2

样例输出

0
4
4
4
1

Solution

考虑\(\min-\max\)容斥,即:

\[\max\{S\}=\sum_{T\subseteq S}(-1)^{|T|+1}\min\{T\}
\]

在这题,\(\min\{S\}\)表示第一次到达\(S\)集合的任何一个点的期望时间,\(\max\{S\}\)同理。

那么我们考虑如何求出\(\min\{S\}\)。

我们\(O(2^n)\)的枚举每一个集合,那么我们考虑如何算。

设\(f(x)\)表示从\(x\)点出发,第一次到达\(S\)的期望时间。

那么我们可以得到一个很显然的式子:

\[f(x)=1+\frac{f(fa)}{d(x)}+ \frac{1}{d(x)}\sum_{v\in son_x}f(v)
\]

那么我们得到了一个\(O(2^nn^3)\)的预处理,显然这是不允许的,我们需要优化。

这里有一个树上期望\(dp\)的套路:\(f(x)\)一定和\(f(fa)\)线性相关,那么我们可以写成\(f(x)=A_xf(fa)+B_x\)。

那么我们可以化一下上面那个式子:

\[f(x)=1+\frac{f(fa)}{d(x)}+ \frac{1}{d(x)}\sum_{v\in son_x}(A_vf(x)+B_v)
\]

移项可得:

\[(1-\frac{\sum_{v\in son}A_v}{d(x)})f(x)=1+\frac{f(fa)}{d(x)}+\frac{\sum_{v\in son B_v}}{d(x)}
\]

\[f(x)=\frac{f(fa)+d(x)+\sum_{v\in son }B_v}{d(x)-\sum_{v\in son}A_v}
\]

对比一下可以得到:

\[A_x=\frac{1}{d(x)-\sum_{v\in son}A_v},B_x=\frac{d(x)+\sum_{v\in son }B_v}{d(x)-\sum_{v\in son}A_v}
\]

那么直接\(dfs\)一遍就可以求出所有的\(A,B\),注意到\(dfs\)到集合中的点就把\(A,B\)设成\(0\)然后\(return\)就好了。

那么我们可以在\(O(2^nn)\)的时间内预处理出\(\min\)。

但是如果这样写还是不能过,因为询问的复杂度上限是\(O(Q2^n)\)。不过据说数据水这样能轻松过?

注意到\(\min-\max\)容斥那个容斥系数只和\(T\)相关,那么我们可以求出\(\min\)之后先乘上一个系数,然后\(fwt\)一下求出子集和,就可以直接\(O(2^n n)\)的预处理出\(\max\)了。

那么对于询问就直接\(O(1)\)输出就好了。

总复杂度\(O((2^n+Q)n)\)。

#include<bits/stdc++.h>
using namespace std; void read(int &x) {
x=0;int f=1;char ch=getchar();
for(;!isdigit(ch);ch=getchar()) if(ch=='-') f=-f;
for(;isdigit(ch);ch=getchar()) x=x*10+ch-'0';x*=f;
} void print(int x) {
if(x<0) putchar('-'),x=-x;
if(!x) return ;print(x/10),putchar(x%10+48);
}
void write(int x) {if(!x) putchar('0');else print(x);putchar('\n');} const int maxn = 19;
const int maxm = (1<<18)+100;
const int mod = 998244353; int head[maxn],tot,n,rt,Q,mn[maxm],f[maxn],A[maxn],B[maxn],d[maxn];
struct edge{int to,nxt;}e[maxn<<1]; int qpow(int a,int x) {
int res=1;
for(;x;x>>=1,a=1ll*a*a%mod) if(x&1) res=1ll*res*a%mod;
return res;
} void ins(int u,int v) {e[++tot]=(edge){v,head[u]},head[u]=tot,d[v]++;} struct data{int a,b;}; data dfs(int x,int fa,int s) {
if(s>>(x-1)&1) return (data){0,0};
int a=0,b=0;data res;
for(int i=head[x];i;i=e[i].nxt)
if(e[i].to!=fa) res=dfs(e[i].to,x,s),a=(a+res.a)%mod,b=(b+res.b)%mod;
res.a=qpow((d[x]-a)%mod,mod-2);
res.b=1ll*res.a*(b+d[x])%mod;
return res;
} void fwt(int *r,int m) {
for(int i=1;i<m;i<<=1)
for(int j=0;j<m;j+=i<<1)
for(int k=0;k<i;k++)
r[i+j+k]=(r[i+j+k]+r[j+k])%mod;
} int k,a[maxn]; int main() {
read(n),read(Q),read(rt);
for(int i=1,x,y;i<n;i++) read(x),read(y),ins(x,y),ins(y,x);
for(int i=1;i<(1<<n);i++) {
mn[i]=dfs(rt,0,i).b;
mn[i]=mn[i]*(__builtin_popcount(i)&1?1:-1);
}
fwt(mn,1<<n);
while(Q--) {
read(k);int res=0;
for(int i=1,x;i<=k;i++) read(x),res|=1<<(x-1);
write((mn[res]+mod)%mod);
}
return 0;
}

[LOJ#2542] [PKUWC2018] 随机游走的更多相关文章

  1. LOJ #2542 [PKUWC2018]随机游走 (概率期望、组合数学、子集和变换、Min-Max容斥)

    很好很有趣很神仙的题! 题目链接: https://loj.ac/problem/2542 题意: 请自行阅读 题解首先我们显然要求的是几个随机变量的最大值的期望(不是期望的最大值),然后这玩意很难求 ...

  2. 【LOJ#2542】[PKUWC2018]随机游走(min-max容斥,动态规划)

    [LOJ#2542][PKUWC2018]随机游走(min-max容斥,动态规划) 题面 LOJ 题解 很明显,要求的东西可以很容易的进行\(min-max\)容斥,那么转为求集合的\(min\). ...

  3. LOJ2542 PKUWC2018 随机游走 min-max容斥、树上高斯消元、高维前缀和、期望

    传送门 那么除了D1T3,PKUWC2018就更完了(斗地主这种全场0分的题怎么会做啊) 发现我们要求的是所有点中到达时间的最大值的期望,\(n\)又很小,考虑min-max容斥 那么我们要求从\(x ...

  4. [PKUWC2018] 随机游走

    Description 给定一棵 \(n\) 个结点的树,你从点 \(x\) 出发,每次等概率随机选择一条与所在点相邻的边走过去. 有 \(Q\) 次询问,每次询问给定一个集合 \(S\),求如果从 ...

  5. 【洛谷5643】[PKUWC2018] 随机游走(Min-Max容斥+待定系数法+高维前缀和)

    点此看题面 大致题意: 从一个给定点出发,在一棵树上随机游走,对于相邻的每个点均有\(\frac 1{deg}\)的概率前往.多组询问,每次给出一个点集,求期望经过多少步能够访问过点集内所有点至少一次 ...

  6. 洛谷 P5643 - [PKUWC2018]随机游走(Min-Max 容斥+FWT+树上高斯消元,hot tea)

    题面传送门 一道挺综合的 hot tea,放到 PKUWC 的 D2T2 还挺喜闻乐见的( 首先我们考虑怎样对一个固定的集合 \(S\) 计算答案,注意到我们要求的是一个形如 \(E(\max(S)) ...

  7. 题解-PKUWC2018 随机游走

    Problem loj2542 题意:一棵 \(n\) 个结点的树,从点 \(x\) 出发,每次等概率随机选择一条与所在点相邻的边走过去,询问走完一个集合 \(S\)的期望时间,多组询问 \(n\le ...

  8. LOJ2542 PKUWC2018随机游走(概率期望+容斥原理)

    如果直接dp,状态里肯定要带上已走过的点的集合,感觉上不太好做. 考虑一种对期望的minmax容斥:其中Max(S)为遍历完S集合的期望步数,Min(S)为遍历到S集合中一个点的期望步数.当然才不管怎 ...

  9. [LOJ2542][PKUWC2018]随机游走(MinMax容斥+树形DP)

    MinMax容斥将问题转化为求x到S中任意点的最小时间. 树形DP,直接求概率比较困难,考虑只求系数.最后由于x节点作为树根无父亲,所以求出的第二个系数就是答案. https://blog.csdn. ...

随机推荐

  1. Java:内存泄露和内存溢出

    1. 内存溢出 (Memory Overflow) 是指程序在申请内存时,没有足够的内存空间供其使用,出现out of memory:比如申请了一个integer,但给它存了long才能存下的数,那就 ...

  2. 【廖雪峰老师python教程】——IO编程

    同步IO 异步IO 最常见的IO——读写文件 读写文件前,我们先必须了解一下,在磁盘上读写文件的功能都是由操作系统提供的,现代操作系统不允许普通的程序直接操作磁盘,所以,读写文件就是请求操作系统打开一 ...

  3. 跟浩哥学自动化测试Selenium -- 浏览器的基本操作与元素定位(3)

    浏览器的基本操作与元素定位 通过上一章学习,我们已经学会了如何设置驱动路径,如何创建浏览器对象,如何打开一个网站,接下来我们要进行一些复杂的操作比如先打开百度首页,在打开博客园,网页后退,前进等等,甚 ...

  4. (C#)工厂方法模式

    1.工厂方法模式 第一了一个用于创建对象的接口,让子类自己决定实例化哪一个类.工厂方法使一个类的实例化延迟到其子类. *工厂方法模式即克服了简单工厂模式违反开放-封闭原则的缺点,又保留了封装对象创建过 ...

  5. 前端开发工程师 - 03.DOM编程艺术 - 期末考试

    期末考试客观题 返回    倒计时: 01:24 1 单选(2分) 以下选项中不是节点类型的是 A. COMMENT_NODE B. DOCUMENT_NODE C. BODY_NODE D. E ...

  6. 转:vue生命周期流程图

  7. 209. First Unique Character in a String

    Description Find the first unique character in a given string. You can assume that there is at least ...

  8. String和StringBuffer以及StringBuilder的区别

    今天在读<java编程思想>的时间,在看到String和StringBuffer以及StringBuffer这三个类的时间,做一个随笔小结,为自己的面试做好准备! 一:String,Str ...

  9. markdown语法介绍

    1. 标题类 每级标题用"# title"表示,共支持6级标题: 2. 段落类 1.建议用换行符控制: 2.用"<p></p>"控制: ...

  10. 【第三章】Shell 变量的数值计算

    一.算数运算符 shell中常见的算术运算符: shell中常见的算术命令: 1. 整数运算 方法一:expr  expr命令就既可以用于整数运算,也可以用于相关字符串长度.匹配等的运算处理: exp ...