「PKUWC2018」Minimax
题面
题解
线段树合并吼题啊
合并的时候要记一下$A$点权值小于$l$的概率和$A$点权值大于$r$的概率,对$B$点同样做
时空复杂度$\text O(nlogw)$,$w$为其中权值的最大值
代码
#include<cstdio>
#include<cstring>
#include<cctype>
#include<algorithm>
#define RG register
#define file(x) freopen(#x".in", "r", stdin);freopen(#x".out", "w", stdout);
#define clear(x, y) memset(x, y, sizeof(x))
inline int read()
{
int data = 0, w = 1; char ch = getchar();
while(ch != '-' && (!isdigit(ch))) ch = getchar();
if(ch == '-') w = -1, ch = getchar();
while(isdigit(ch)) data = data * 10 + (ch ^ 48), ch = getchar();
return data * w;
}
const int Mod(998244353), Inv(796898467), maxn(300010), LIM((1 << 30) - 1);
inline int Add(int a, int b) { return (a + b) % Mod; }
int ans, cnt, son[2][maxn * 20], tag[maxn * 20], P[maxn * 20], cur, root[maxn];
inline int newNode()
{
tag[++cur] = 1; P[cur] = 0;
son[0][cur] = son[1][cur] = 0;
return cur;
}
inline void pushup(int x) { P[x] = Add(P[son[0][x]], P[son[1][x]]); }
inline void put_tag(int x, int p)
{
if(!x) return;
tag[x] = 1ll * tag[x] * p % Mod;
P[x] = 1ll * P[x] * p % Mod;
}
inline void pushdown(int x)
{
put_tag(son[0][x], tag[x]);
put_tag(son[1][x], tag[x]);
tag[x] = 1;
}
void calc(int x, int l = 0, int r = LIM)
{
if(!x) return;
if(l == r)
{
++cnt; ans = Add(ans, 1ll * cnt * P[x] % Mod * P[x] % Mod * l % Mod);
return;
}
pushdown(x);
int mid = (l + r) >> 1;
calc(son[0][x], l, mid);
calc(son[1][x], mid + 1, r);
}
void insert(int &x, int id, int l = 0, int r = LIM)
{
if(!x) x = newNode();
tag[x] = P[x] = 1;
if(l == r) return;
int mid = (l + r) >> 1;
if(id <= mid) insert(son[0][x], id, l, mid);
else insert(son[1][x], id, mid + 1, r);
}
int S[2][maxn], tot[maxn], W[maxn], n;
int merge(int a, int b, int pa, int pb, int pmax)
{
if(!a && !b) return 0;
if(!a) return put_tag(b, pa), b;
if(!b) return put_tag(a, pb), a;
pushdown(a), pushdown(b);
int pal = Add(pa, 1ll * P[son[1][a]] * Add(Mod - pmax, 1) % Mod),
pbl = Add(pb, 1ll * P[son[1][b]] * Add(Mod - pmax, 1) % Mod),
par = Add(pa, 1ll * P[son[0][a]] * pmax % Mod),
pbr = Add(pb, 1ll * P[son[0][b]] * pmax % Mod);
son[0][a] = merge(son[0][a], son[0][b], pal, pbl, pmax);
son[1][a] = merge(son[1][a], son[1][b], par, pbr, pmax);
return pushup(a), a;
}
void dfs(int x)
{
if(!tot[x]) return (void) (insert(root[x], W[x]));
W[x] = 1ll * W[x] * Inv % Mod;
for(RG int i = 0; i < tot[x]; i++) dfs(S[i][x]);
if(tot[x] == 1) root[x] = root[S[0][x]];
else root[x] = merge(root[S[0][x]], root[S[1][x]], 0, 0, W[x]);
}
int main()
{
#ifndef ONLINE_JUDGE
file(cpp);
#endif
n = read();
for(RG int i = 1, fa; i <= n; i++)
fa = read(), S[tot[fa]++][fa] = i;
for(RG int i = 1; i <= n; i++) W[i] = read();
dfs(1); calc(root[1]); printf("%d\n", ans);
return 0;
}
「PKUWC2018」Minimax的更多相关文章
- loj#2537. 「PKUWC2018」Minimax
题目链接 loj#2537. 「PKUWC2018」Minimax 题解 设\(f_{u,i}\)表示选取i的概率,l为u的左子节点,r为u的子节点 $f_{u,i} = f_{l,i}(p \sum ...
- LOJ2537. 「PKUWC2018」Minimax【概率DP+线段树合并】
LINK 思路 首先暴力\(n^2\)是很好想的,就是把当前节点概率按照权值大小做前缀和和后缀和然后对于每一个值直接在另一个子树里面算出贡献和就可以了,注意乘上选最大的概率是小于当前权值的部分,选最小 ...
- loj2537 「PKUWC2018」Minimax 【概率 + 线段树合并】
题目链接 loj2537 题解 观察题目的式子似乎没有什么意义,我们考虑计算出每一种权值的概率 先离散化一下权值 显然可以设一个\(dp\),设\(f[i][j]\)表示\(i\)节点权值为\(j\) ...
- 【LOJ】#2537. 「PKUWC2018」Minimax
题解 加法没写取模然后gg了QwQ,de了半天 思想还是比较自然的,线段树合并的维护方法我是真的很少写,然后没想到 很显然,我们有个很愉快的想法是,对于每个节点枚举它所有的叶子节点,对于一个叶子节点的 ...
- LOJ2537. 「PKUWC2018」Minimax [DP,线段树合并]
传送门 思路 首先有一个\(O(n^2)\)的简单DP:设\(dp_{x,w}\)为\(x\)的权值为\(w\)的概率. 假设\(w\)来自\(v1\)的子树,那么有 \[ dp_{x,w}=dp_{ ...
- Loj #2542. 「PKUWC2018」随机游走
Loj #2542. 「PKUWC2018」随机游走 题目描述 给定一棵 \(n\) 个结点的树,你从点 \(x\) 出发,每次等概率随机选择一条与所在点相邻的边走过去. 有 \(Q\) 次询问,每次 ...
- 「PKUWC2018」随机游走(min-max容斥+FWT)
「PKUWC2018」随机游走(min-max容斥+FWT) 以后题目都换成这种「」形式啦,我觉得好看. 做过重返现世的应该看到就想到 \(min-max\) 容斥了吧. 没错,我是先学扩展形式再学特 ...
- 「PKUWC2018」猎人杀
「PKUWC2018」猎人杀 解题思路 首先有一个很妙的结论是问题可以转化为已经死掉的猎人继续算在概率里面,每一轮一直开枪直到射死一个之前没死的猎人为止. 证明,设所有猎人的概率之和为 \(W\) , ...
- LOJ3044. 「ZJOI2019」Minimax 搜索
LOJ3044. 「ZJOI2019」Minimax 搜索 https://loj.ac/problem/3044 分析: 假设\(w(1)=W\),那么使得这个值变化只会有两三种可能,比\(W\)小 ...
随机推荐
- Java并发案例03---生产者消费者问题02
生产者消费者第二种情形 package com.maple.msb.one; public class ProducerConsumer { public static void main(Strin ...
- PAT——1050. 螺旋矩阵
本题要求将给定的N个正整数按非递增的顺序,填入“螺旋矩阵”.所谓“螺旋矩阵”,是指从左上角第1个格子开始,按顺时针螺旋方向填充.要求矩阵的规模为m行n列,满足条件:m*n等于N:m>=n:且m- ...
- Dynamic Rankings—带单点修改的主席树
这道题应该是很不错的板子了\(\mathcal{\color{cyan}{Link}}\) \(\mathcal{\color{red}{Description}}\) 给定一个序列,有两种操作.一是 ...
- Linux下jmap命令查看内存使用
Linux下jmap命令查看内存使用 jmap -heap 1234(1234为进程号) jmap是JDK自带的一个工具,非常小巧方便,其支持参数如下: -heap 打印heap空间的概要 ...
- HomeKit 开发指南(中文版)
转载自cocoachina 本文由CocoaChina翻译组成员iBenjamin_Go和浅夏@旧时光翻译自苹果开发文档:HomeKit Developer Guide,敬请勘误. 本文档内容包括 第 ...
- LWIP network interface 即 LWIP 的 硬件 数据 接口 移植 详解 STM32 以太网数据 到达 的第二站: void ethernetif_input( void * pvParameters )
根据 上一篇 文章 , ETH DMA 数据中断 会 发送 一个信号量 ,我使用 全局 搜索 这个信号量 s_xSemaphore 得到 一下 几个 值 根据 这个 分析 我们找到了 数据 的 ...
- sharepoint OOS打开文档使用新窗口
总体说来就是在<a>标签上嵌入_blank $("a[onclick*='return DispEx'][target!='_blank']") .attr(" ...
- BUAA_OO第二单元总结性博客作业——多线程电梯架构
一.设计策略 在第一次作业时,我刚第一次接触多线程这个东西……于是乎对于第一次VIP直上直下一次只接一个人的电梯,我借鉴了指导书中为我们提供的架构,设计了一个输入线程和一个电梯线程,并设置了一个中间类 ...
- C# 删除指定目录下的所有文件及文件夹
using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Tex ...
- JS变量、作用域、内存
写到这个题目<JS变量.作用域,内存>,我就不由自主想起了黄金三嫖客.可能是名字有点像,嗯,一定是这样子的! JS接触下来,应该是要比Java简单不少的,所以,要学好啊.立个flag半年后 ...