loj2537 「PKUWC2018」Minimax 【概率 + 线段树合并】
题目链接
题解
观察题目的式子似乎没有什么意义,我们考虑计算出每一种权值的概率
先离散化一下权值
显然可以设一个\(dp\),设\(f[i][j]\)表示\(i\)节点权值为\(j\)的概率
如果\(i\)是叶节点显然
如果\(i\)只有一个儿子直接继承即可
如果\(i\)有两个儿子,对于儿子\(x\),设另一个儿子为\(y\)
则有
\]
直接转移是\(O(n^2)\)的,发现每个节点都有\(O(n)\)个位置需要转移
考虑优化,可以考虑线段树合并
对于一个子树中的权值\(x\),我们记另一棵子树比它大的概率为\(maxa\),
则\(x\)的概率要乘上\(maxa(1 - p_i) + (1 - maxa)p_i = maxa + p_i - 2p_imaxa\)
所以我们在线段树合并过程中,优先合并右子树,并更新两棵子树的\(maxa\)与\(maxb\),就可以在合并过程中转移了
复杂度\(O(nlogn)\)
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<map>
#define Redge(u) for (int k = h[u],to; k; k = ed[k].nxt)
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define mp(a,b) make_pair<int,int>(a,b)
#define cls(s) memset(s,0,sizeof(s))
#define cp pair<int,int>
#define LL long long int
using namespace std;
const int maxn = 300005,maxm = 8000005,INF = 1000000000,P = 998244353;
inline int read(){
int out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57){if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57){out = (out << 3) + (out << 1) + c - 48; c = getchar();}
return out * flag;
}
int n,Ls[maxn],Rs[maxn],b[maxn],N,v10000;
int rt[maxn],sum[maxm],ls[maxm],rs[maxm],tag[maxm],cnt;
int p[maxn],maxa,maxb;
inline int qpow(int a,int b){
int re = 1;
for (; b; b >>= 1,a = 1ll * a * a % P)
if (b & 1) re = 1ll * re * a % P;
return re;
}
inline void pd(int u){
if (tag[u] > 1){
sum[ls[u]] = 1ll * sum[ls[u]] * tag[u] % P;
sum[rs[u]] = 1ll * sum[rs[u]] * tag[u] % P;
tag[ls[u]] = 1ll * tag[ls[u]] * tag[u] % P;
tag[rs[u]] = 1ll * tag[rs[u]] * tag[u] % P;
tag[u] = 1;
}
}
void modify(int& u,int l,int r,int pos){
u = ++cnt; sum[u] = tag[u] = 1;
if (l == r) return;
int mid = l + r >> 1;
if (mid >= pos) modify(ls[u],l,mid,pos);
else modify(rs[u],mid + 1,r,pos);
}
int merge(int u,int v,int p){
if (!u && !v) return 0;
if (!u){
maxb = (maxb + sum[v]) % P;
int tmp;
tmp = (((maxa + p) % P - 2ll * p * maxa % P) % P + P) % P;
sum[v] = 1ll * sum[v] * tmp % P;
tag[v] = 1ll * tag[v] * tmp % P;
return v;
}
if (!v){
maxa = (maxa + sum[u]) % P;
int tmp;
tmp = (((maxb + p) % P - 2ll * p * maxb % P) % P + P) % P;
sum[u] = 1ll * sum[u] * tmp % P;
tag[u] = 1ll * tag[u] * tmp % P;
return u;
}
pd(u); pd(v);
int t = ++cnt; tag[t] = 1;
rs[t] = merge(rs[u],rs[v],p);
ls[t] = merge(ls[u],ls[v],p);
sum[t] = (sum[ls[t]] + sum[rs[t]]) % P;
return t;
}
void dfs(int u){
if (!Ls[u]) modify(rt[u],1,N,p[u]);
else if (!Rs[u]) dfs(Ls[u]),rt[u] = rt[Ls[u]];
else {
dfs(Ls[u]); dfs(Rs[u]);
maxa = maxb = 0;
rt[u] = merge(rt[Ls[u]],rt[Rs[u]],p[u]);
}
}
int ans;
void cal(int u,int l,int r){
if (l == r) {ans = (ans + 1ll * l * b[l] % P * sum[u] % P * sum[u] % P) % P;return;}
pd(u);
int mid = l + r >> 1;
cal(ls[u],l,mid);
cal(rs[u],mid + 1,r);
}
int main(){
n = read(); read(); int x; v10000 = qpow(10000,P - 2);
for (int i = 2; i <= n; i++){
x = read();
if (!Ls[x]) Ls[x] = i;
else Rs[x] = i;
}
for (int i = 1; i <= n; i++){
p[i] = read();
if (!Ls[i]) b[++N] = p[i];
else p[i] = 1ll * p[i] * v10000 % P;
}
sort(b + 1,b + 1 + N);
for (int i = 1; i <= n; i++)
if (!Ls[i]) p[i] = lower_bound(b + 1,b + 1 + N,p[i]) - b;
dfs(1);
cal(rt[1],1,N);
printf("%d\n",ans);
return 0;
}
loj2537 「PKUWC2018」Minimax 【概率 + 线段树合并】的更多相关文章
- LOJ2537. 「PKUWC2018」Minimax【概率DP+线段树合并】
LINK 思路 首先暴力\(n^2\)是很好想的,就是把当前节点概率按照权值大小做前缀和和后缀和然后对于每一个值直接在另一个子树里面算出贡献和就可以了,注意乘上选最大的概率是小于当前权值的部分,选最小 ...
- LOJ2537. 「PKUWC2018」Minimax [DP,线段树合并]
传送门 思路 首先有一个\(O(n^2)\)的简单DP:设\(dp_{x,w}\)为\(x\)的权值为\(w\)的概率. 假设\(w\)来自\(v1\)的子树,那么有 \[ dp_{x,w}=dp_{ ...
- 「洛谷4197」「BZOJ3545」peak【线段树合并】
题目链接 [洛谷] [BZOJ]没有权限号嘤嘤嘤.题号:3545 题解 窝不会克鲁斯卡尔重构树怎么办??? 可以离线乱搞. 我们将所有的操作全都存下来. 为了解决小于等于\(x\)的操作,那么我们按照 ...
- loj#2537. 「PKUWC2018」Minimax
题目链接 loj#2537. 「PKUWC2018」Minimax 题解 设\(f_{u,i}\)表示选取i的概率,l为u的左子节点,r为u的子节点 $f_{u,i} = f_{l,i}(p \sum ...
- 「CQOI2006」简单题 线段树
「CQOI2006」简单题 线段树 水.区间修改,单点查询.用线段树维护区间\([L,R]\)内的所有\(1\)的个数,懒标记表示为当前区间是否需要反转(相对于区间当前状态),下方标记时懒标记取反即可 ...
- BZOJ.5461.[PKUWC2018]Minimax(DP 线段树合并)
BZOJ LOJ 令\(f[i][j]\)表示以\(i\)为根的子树,权值\(j\)作为根节点的概率. 设\(i\)的两棵子树分别为\(x,y\),记\(p_a\)表示\(f[x][a]\),\(p_ ...
- 【LOJ】#2537. 「PKUWC2018」Minimax
题解 加法没写取模然后gg了QwQ,de了半天 思想还是比较自然的,线段树合并的维护方法我是真的很少写,然后没想到 很显然,我们有个很愉快的想法是,对于每个节点枚举它所有的叶子节点,对于一个叶子节点的 ...
- 「PKUWC2018」Minimax
题面 题解 强势安利一波巨佬的$blog$ 线段树合并吼题啊 合并的时候要记一下$A$点权值小于$l$的概率和$A$点权值大于$r$的概率,对$B$点同样做 时空复杂度$\text O(nlogw)$ ...
- [PKUWC2018]Minimax [dp,线段树合并]
好妙的一个题- 我们设 \(f_{i,j}\) 为 \(i\) 节点出现 \(j\) 的概率 设 \(l = ch[i][0] , r = ch[i][1]\) 即左儿子右儿子 设 \(m\) 为叶子 ...
随机推荐
- Python模块搜索路径
当一个名为 spam 的模块被导入的时候,解释器首先寻找具有该名称的内置模块.如果没有找到,然后解释器从 sys.path 变量给出的目录列表里寻找名为 spam.py 的文件.sys.path 初始 ...
- FFM原理及公式推导
原文来自:博客园(华夏35度)http://www.cnblogs.com/zhangchaoyang 作者:Orisun 上一篇讲了FM(Factorization Machines),说一说FFM ...
- 如何掌握 Kubernetes ?系统学习 k8s 的大纲一份
深度剖析 Kubernetes 深度剖析 k8s 如何学习 Kubernetes ?如何入门 Kubernetes? 为了帮帮初学者,2018 年 InfoQ 旗下(就是你知道的那个 InfoQ 哇) ...
- UI设计学习笔记(7-12)
UI学习笔记(7)--扁平化图标 认识扁平化 Flat Design 抛弃传统的渐变.阴影.高光等拟真视觉效果,打造看上去更平的界面.(颜色.形状) 扁平化图标有什么优缺点 优点: 简约不简单.有新鲜 ...
- Go单元测试注意事项及测试单个方法和整个文件的命令
Go程序开发过程中免不了要对所写的单个业务方法进行单元测试,Go提供了 "testing" 包可以实现单元测试用例的编写,不过想要正确编写单元测试需要注意以下三点: Go文件名必须 ...
- Python参数传递,既不是传值也不是传引用
面试的时候,有没有被问到Python传参是传引用还是传值这种问题?有没有听到过Python传参既不是传值也不是传引用这种说法?一个小小的参数默认值也可能让代码出现难以查找的bug? 如果你也遇到过上面 ...
- Vue 实例详解与生命周期
Vue 实例详解与生命周期 Vue 的实例是 Vue 框架的入口,其实也就是前端的 ViewModel,它包含了页面中的业务逻辑处理.数据模型等,当然它也有自己的一系列的生命周期的事件钩子,辅助我们进 ...
- Scrum Meeting 2 -2014.11.2
今天大家读完代码后又聚在了一块讨论了许多.确定了重点的任务和分工细节.提出了许多问题和改进的方案.还有讨论分析了关于团队作业 - 软件分析和用户需求调查,初步决定目标软件为必应的输入法和词典,团队为争 ...
- Daily Scrum (2015/11/3)
今天我们的爬虫能在pc上成功运行并且把所爬取的数据存到服务器上了!我们已经搭建好数据库,把相关信息存到数据库中,并把数据存到D盘里共享给数据处理小组使用. 成员 今日工作 时间 明日工作 符美潇 完成 ...
- Leetcode题库——16.最接近的三数之和
@author: ZZQ @software: PyCharm @file: threeSumClosest.py @time: 2018/10/14 20:28 说明:最接近的三数之和. 给定一个包 ...