根据题意,若一个点有子节点,则给出权值;否则可以从子节点转移得来。

若没有子节点,则直接给出权值;

若只有一个子节点,则概率情况与该子节点完全相同;

若有两个子节点,则需要从两个子节点中进行转移。

如何转移?显然,若权值 $i$ 在左子树,要取到它,需要在 $p_i$ 的概率中左子树较大,在 $(1-p_i)$ 的概率中左子树较小,右子树同理。因为当权值 $i$ 在左子树时右子树取到它的概率为 $0$ ,因此可以直接将两个子树的转移式相加合并,没有影响。即:

设节点 $x$ 取到权值 $i$ 的概率为 $f_{x,i}$ ,节点数为 $n$ ,则有:

$$f_{x,i}=f_{lson(x),i}*(p_x*\sum_{j=1}^{i-1} f_{rson(x),j}+(1-p_x)*\sum_{j=i+1}^{n} f_{rson(x),j})+f_{rson(x),i}*(p_x*\sum_{j=1}^{i-1} f_{lson(x),j}+(1-p_x)*\sum_{j=i+1}^{n} f_{lson(x),j})$$

如何求值?通过观察可以发现,这个式子同时需要左儿子的值、右儿子的值以及其前缀、后缀和,可以想到用线段树合并进行求值。

如何实现?在线段树合并的同时维护前、后缀和,打上乘法标记即可。记得离散化。

本题需要用到分数取模:$\frac{a}{b}\bmod k=\frac{a}{b}*b^{k-1}\bmod k=a*b^{k-2}\bmod k$ 。

#include<iostream>
#include<cstdio>
#include<algorithm>
using namespace std;
const int N=3e5+100;
const int P=998244353;
int qpow(int x,int y)
{
int ans=1;
while(y)
{
if(y&1)
ans=1LL*ans*x%P;
x=1LL*x*x%P,y>>=1;
}
return ans;
}//快速幂
const int D=qpow(10000,P-2);
struct Seg
{
int lson,rson,sum,tag;
#define lson(i) t[i].lson
#define rson(i) t[i].rson
#define sum(i) t[i].sum
#define tag(i) t[i].tag//乘法标记
}t[N<<5];
int n,cnt,h,ans;
int v[N],r[N],a[N],cnts[N],anss[N],s[N][2];
int newp()
{
int p=++cnt;
tag(p)=1;
return p;
}//新建节点
void update(int p,int val)
{
sum(p)=1LL*sum(p)*val%P;
tag(p)=1LL*tag(p)*val%P;
}//更新
void pushdown(int p)
{
if(tag(p)>1)
update(lson(p),tag(p)),update(rson(p),tag(p)),tag(p)=1;
}//下传标记
void change(int &p,int l,int r,int k,int val)
{
if(!p)
p=newp();
if(l==r)
{
sum(p)=val;
return ;
}
pushdown(p);
int mid=l+r>>1;
if(k<=mid)
change(lson(p),l,mid,k,val);
else
change(rson(p),mid+1,r,k,val);
sum(p)=sum(lson(p))+sum(rson(p))%P;
}//修改
void ask(int p,int l,int r)
{
if(!p)
return ;
if(l==r)
{
anss[l]=sum(p);
return ;
}
pushdown(p);
int mid=l+r>>1;
ask(lson(p),l,mid),ask(rson(p),mid+1,r);
}//输出答案到对应数组
int merge(int x,int y,int l,int r,int xtag,int ytag,int val)
{
if(!x && !y)
return 0;
if(!x)
{
update(y,ytag);
return y;
}
if(!y)
{
update(x,xtag);
return x;
}
pushdown(x),pushdown(y);//下传标记
int mid=l+r>>1,lx=sum(lson(x)),ly=sum(lson(y)),rx=sum(rson(x)),ry=sum(rson(y));//先存值,否则之后会被改动
lson(x)=merge(lson(x),lson(y),l,mid,(xtag+1LL*ry*(1-val+P)%P)%P,(ytag+1LL*rx*(1-val+P)%P)%P,val);
rson(x)=merge(rson(x),rson(y),mid+1,r,(xtag+1LL*ly*val%P)%P,(ytag+1LL*lx*val%P)%P,val);
sum(x)=(sum(lson(x))+sum(rson(x)))%P;
return x;
}
void pre()
{
sort(a+1,a+h+1);
for(int i=1;i<=n;i++)
if(!cnts[i])
v[i]=lower_bound(a+1,a+h+1,v[i])-a;//权值离散化
else
v[i]=1LL*v[i]*D%P;//存概率,分数取模
}//预处理权值
void solve(int x)
{
if(!cnts[x])
{
change(r[x],1,h,v[x],1);
return ;
}//没有子节点,插入权值
if(cnts[x]==1)
{
solve(s[x][0]);
r[x]=r[s[x][0]];
return ;
}//只有一个子节点
solve(s[x][0]),solve(s[x][1]);
r[x]=merge(r[s[x][0]],r[s[x][1]],1,h,0,0,v[x]);//有两个子节点
}
int main()
{
scanf("%d",&n);
for(int i=1,x;i<=n;i++)
{
scanf("%d",&x);
if(x)
s[x][cnts[x]++]=i;
}
for(int i=1,x;i<=n;i++)
{
scanf("%d",&v[i]);
if(!cnts[i])
a[++h]=v[i];
}
pre(),solve(1),ask(r[1],1,h);
for(int i=1;i<=h;i++)
ans=(ans+1LL*i*a[i]%P*anss[i]%P*anss[i])%P;//计算答案
printf("%d",ans);
return 0;
}

[PKUWC2018]Minimax 题解的更多相关文章

  1. LOJ2537:[PKUWC2018]Minimax——题解

    https://loj.ac/problem/2537 参考了本题在网上能找到的为数不多的题解. 以及我眼睛瞎没看到需要离散化,还有不开longlong见祖宗. ——————————————————— ...

  2. 【题解】PKUWC2018简要题解

    [题解]PKUWC2018简要题解 Minimax 定义结点x的权值为: 1.若x没有子结点,那么它的权值会在输入里给出,保证这类点中每个结点的权值互不相同. 2.若x有子结点,那么它的权值有p的概率 ...

  3. BZOJ5461: [PKUWC2018]Minimax

    BZOJ5461: [PKUWC2018]Minimax https://lydsy.com/JudgeOnline/problem.php?id=5461 分析: 写出\(dp\)式子:$ f[x] ...

  4. 题解-PKUWC2018 Minimax

    Problem loj2537 Solution pkuwc2018最水的一题,要死要活调了一个多小时(1h59min) 我写这题不是因为它有多好,而是为了保持pkuwc2018的队形,与这题类似的有 ...

  5. [PKUWC2018] Minimax

    Description 给定一棵 \(n\) 个节点的树,每个节点最多有两个子节点. 如果 \(x\) 是叶子,则给定 \(x\) 的权值:否则,它的权值有 \(p_x\) 的概率是它子节点中权值的较 ...

  6. BZOJ.5461.[PKUWC2018]Minimax(DP 线段树合并)

    BZOJ LOJ 令\(f[i][j]\)表示以\(i\)为根的子树,权值\(j\)作为根节点的概率. 设\(i\)的两棵子树分别为\(x,y\),记\(p_a\)表示\(f[x][a]\),\(p_ ...

  7. LOJ2537 PKUWC2018 Minimax 树形DP、线段树合并

    传送门 题意:自己去看 首先可以知道,每一个点都有几率被选到,所以$i$与$V_i$的关系是确定了的. 所以我们只需要考虑每一个值的取到的概率. 很容易设计出一个$DP$:设$f_{i,j}$为在第$ ...

  8. [BZOJ5461][LOJ#2537[PKUWC2018]Minimax(概率DP+线段树合并)

    还是没有弄清楚线段树合并的时间复杂度是怎么保证的,就当是$O(m\log n)$吧. 这题有一个显然的DP,dp[i][j]表示节点i的值为j的概率,转移时维护前缀后缀和,将4项加起来就好了. 这个感 ...

  9. 【洛谷5298】[PKUWC2018] Minimax(树形DP+线段树合并)

    点此看题面 大致题意: 有一棵树,给出每个叶节点的点权(互不相同),非叶节点\(x\)至多有两个子节点,且其点权有\(p_x\)的概率是子节点点权较大值,有\(1-p_x\)的概率是子节点点权较小值. ...

随机推荐

  1. 女生学Java编程是什么感受?

    那我就代表女生来说说感受 在编程的世界很难遇到好看的帅哥 记得当年15年7月4号是我实习生入职的日子,因为是校企合作,所以没有面试.老师推荐.直接入职.刚来北京第一个感觉就是人多,还有就是热.刚到公司 ...

  2. lamt环境搭建

    目录 lamt环境搭建 安装apache 安装mysql 安装tomcat 修改配置文件 lamt环境搭建 环境说明: 系统 IP 需要安装的服务 centos7 192.168.32.125 htt ...

  3. Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:testCompile (default-testCompile) on project docker_springcloud_demo: Fatal error compiling: 无效的标记: -parameters -> [Help 1]

    [ERROR] Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:testCompile (def ...

  4. 【源码】Python3使用Requests抓取和检测电光代理API,并查询ip代理是否成功

    电光代理成立后,做一篇笔记,记录我使用Requests抓取和测试电光代理的方法 很多人学习python,不知道从何学起.很多人学习python,掌握了基本语法过后,不知道在哪里寻找案例上手.很多已经做 ...

  5. 将vscode打造成强大的C/C++ IDE

    一.安装 你可以直接从微软官网下载,如果你想要一个纯净的vscode(微软官方的有一项商标.一个插件库.一个 C# 调试器以及遥测),可以手动编译https://github.com/microsof ...

  6. Flutter build apk 如何访问网络

    将下列配置放到路径:your_project\android\app\src下的 main 文件夹下的 AndroidManifest.xml 和 profile 文件夹下的 AndroidManif ...

  7. .Net 桌面程序(winform,wpf,跨平台avalonia)打安装包部署到windows 入门

    .Net 桌面程序(winform,wpf,跨平台avalonia)部署到windows 入门 本文以为avalonia为例,用Setup Factory 将.Net桌面程序(winform,wpf, ...

  8. NLTK库WordNet的使用方法实例

    1.在代码中引入wordnet包 >>>from nltk.corpus import wordnet as wn 2.查询一个词所在的所有词集(synsets) >>& ...

  9. asp.net core跨平台--CentOS7.2部署asp.net core网站

    随着vs2015 2017的发布,.NETCore越来越流行了,我就尝试着做了个demo,在centos上试着运行了一下,中间遇到很多问题,不过最后还是成功运行,记录一下过程.废话不多说,直接开始: ...

  10. Provisional headers are shown 问题的一种情况

    Provisional headers are shown 出现在请求头的报错里面,意思为 显示临时的头部,真实的意思是,请求没有收到服务器返回.如果出现类似情况, 可以在服务端找一找,是否没有给该请 ...