zoukankan      html  css  js  c++  java
  • 「PKUWC2018」Minimax

    题面

    题解

    考虑一个数字被取到最小值的概率怎么算。

    由于一个节点最多只有 (2) 个儿子,所以 (x) 出现的概率 (a_x) 分为两个部分,一个作为最大值,另一个即作为最小值。

    以计算这个点作为最小值出现的概率为例,这个概率就是这个数在这棵子树内出现的概率 (a_x') 乘以另外一棵子树中取到比它大的数字的概率再乘上这个点取最小值的概率 (1 - p_x)。最大值同理。

    于是对每个点维护一棵线段树维护每个点出现的概率,线段树合并即可。

    代码

    #include <cstdio>
    #include <algorithm>
    #include <vector>
    
    inline int read()
    {
    	int data = 0, w = 1; char ch = getchar();
    	while (ch != '-' && (ch < '0' || ch > '9')) ch = getchar();
    	if (ch == '-') w = -1, ch = getchar();
    	while (ch >= '0' && ch <= '9') data = data * 10 + (ch ^ 48), ch = getchar();
    	return data * w;
    }
    
    const int N(3e5 + 10), Mod(998244353), Inv(796898467), LIM(1e9);
    struct edge { int next, to; } e[N << 1];
    int n, m, w[N], head[N], e_num, rt[N], cnt;
    int cur, son[2][N * 20], tag[N * 20], p[N * 20], ans;
    inline void add_edge(int from, int to)
    {
    	e[++e_num] = (edge) {head[from], to};
    	head[from] = e_num;
    }
    
    void pushup(int x) { p[x] = (p[son[0][x]] + p[son[1][x]]) % Mod; }
    void pushtag(int x, int _)
    	{ p[x] = 1ll * p[x] * _ % Mod, tag[x] = 1ll * tag[x] * _ % Mod; }
    void pushdown(int x)
    {
    	if (tag[x] == 1) return;
    	pushtag(son[0][x], tag[x]);
    	pushtag(son[1][x], tag[x]);
    	tag[x] = 1;
    }
    
    void Solve(int x, int l = 1, int r = LIM)
    {
    	if (!x) return;
    	if (l == r) return (void) (ans = (ans + 1ll * (++cnt) * l % Mod * p[x] % Mod * p[x]) % Mod);
    	int mid = (l + r) >> 1; pushdown(x);
    	Solve(son[0][x], l, mid), Solve(son[1][x], mid + 1, r);
    }
    
    void Insert(int &x, int t, int l = 1, int r = LIM)
    {
    	if (!x) x = ++cur; p[x] = tag[x] = 1;
    	if (l == r) return; int mid = (l + r) >> 1;
    	if (t <= mid) Insert(son[0][x], t, l, mid);
    	else Insert(son[1][x], t, mid + 1, r);
    	pushup(x);
    }
    
    int Merge(int x, int y, int px, int py, int t)
    {
    	if (!x && !y) return 0;
    	if (!x) return pushtag(y, px), y;
    	if (!y) return pushtag(x, py), x;
    	pushdown(x), pushdown(y);
    	int pxl, pxr, pyl, pyr;
    	pxl = (px + 1ll * (Mod + 1 - t) * p[son[1][x]]) % Mod;
    	pyl = (py + 1ll * (Mod + 1 - t) * p[son[1][y]]) % Mod;
    	pxr = (px + 1ll * t * p[son[0][x]]) % Mod;
    	pyr = (py + 1ll * t * p[son[0][y]]) % Mod;
    	son[0][x] = Merge(son[0][x], son[0][y], pxl, pyl, t);
    	son[1][x] = Merge(son[1][x], son[1][y], pxr, pyr, t);
    	return pushup(x), x;
    }
    
    void dfs(int x)
    {
    	int a[2], tot = 0;
    	for (int i = head[x]; i; i = e[i].next)
    		dfs(e[i].to), a[tot++] = rt[e[i].to];
    	if (tot == 0) Insert(rt[x], w[x]);
    	else if (tot == 1) rt[x] = a[0];
    	else rt[x] = Merge(a[0], a[1], 0, 0, 1ll * w[x] * Inv % Mod);
    }
    
    int main()
    {
    	n = read();
    	for (int i = 1; i <= n; i++) add_edge(read(), i);
    	for (int i = 1; i <= n; i++) w[i] = read();
    	dfs(1), Solve(rt[1]), printf("%d
    ", ans);
    	return 0;
    }
    
  • 相关阅读:
    CentOS 7 安装java 环境
    CentOS 7 替换网易yum 源
    九度:题目1553:时钟
    Maximum Subarray
    职场细节
    poj2524 Ubiquitous Religions
    九度 1526:朋友圈
    程序载入
    设备管理
    操作系统系列
  • 原文地址:https://www.cnblogs.com/cj-xxz/p/11979679.html
Copyright © 2011-2022 走看看