\(\mathcal{Description}\)

  Link.

  在 NOIP 2020 A 的基础上,每条边赋权值 \(a_i\),随机恰好一条边断掉,第 \(i\) 条段的概率正比于 \(a_i\)。求每个汇集口收集到污水的期望吨数。答案模 \(998244353\)(我谢谢出题人。

\(\mathcal{Solution}\)

  方法一 这个题麻烦的地方在于 DAG 上断边,很难将每条断边的贡献一起计算(注意不是“叠加”,仅仅是一下子算出分别断开多条边的贡献之和)。我们得想个办法保持 DAG 的静态结构不变

  考虑若 \(\lang u,v\rang\) 断开,那么到达 \(u\) 的污水量期望 \(x\) 是已知的,且对于除了 \(v\) 之外 \(u\) 的所有后继 \(w\),流入量都会增加 \(\Delta_1=\frac{x}{d_u(d_u-1)}\);\(v\) 的流入量会减少 \(\Delta_2=\frac{x}{d_u}\)。当然“所有后继”所含要素过多,我们给它放到 \(u\) 身上,结合 \(\Delta_1,\Delta_2\),重新描述一下断边的影响:

  断掉 \(\lang u,v\rang\),\(u\) 会获得一条私人流入管道,流入量 \(I_u=\frac{x}{d_u-1}\);\(v\) 会获得一条私人流出管道,流出量 \(O_v=\frac{x}{d_u(d_u+1)}+\frac{x}{d_u}\)。注意 \(\lang u,v\rang\) 仍然存在且具有正常运输功能。

  嗯,DAG 不动了,随便算叭。


  方法二 \(\sf OneInDark\) 太厉害辣!

\(f(i,...)\),表示拓扑序前 \(i\) 个构成的 DAG 中,……

  不要被该死的树 DP 限制了,不要一直去想构造子树结构。

  两个方法复杂度都是 \(\mathcal O(n+m)\)。

\(\mathcal{Code}\)

  方法一。

/*+Rainybunny+*/

#include <bits/stdc++.h>

#define rep(i, l, r) for (int i = l, rep##i = r; i <= rep##i; ++i)
#define per(i, r, l) for (int i = r, per##i = l; i >= per##i; --i) inline char fgc() {
static char buf[1 << 17], *p = buf, *q = buf;
return p == q && (q = buf + fread(p = buf, 1, 1 << 17, stdin), p == q)
? EOF : *p++;
} inline int rint() {
int x = 0, s = fgc();
for (; s < '0' || '9' < s; s = fgc());
for (; '0' <= s && s <= '9'; s = fgc()) x = x * 10 + (s ^ '0');
return x;
} inline void wint(const int x) {
if (9 < x) wint(x / 10);
putchar(x % 10 ^ '0');
} const int MAXN = 2e5, MAXM = 5e5, MOD = 998244353;
int n, A, B, m, sum, ecnt, head[MAXN + 5];
int ideg[MAXN + 5], odeg[MAXN + 5], osum[MAXN + 5];
int inv[MAXM + 5], que[MAXN + 5], hd, tl, f[MAXN + 5], g[MAXN + 5];
struct Edge { int to, old, nxt; } graph[MAXM + 5]; inline void link(const int s, const int t, const int o) {
graph[++ecnt] = { t, o, head[s] }, head[s] = ecnt;
} inline int mul(const int u, const int v) { return 1ll * u * v % MOD; }
inline void subeq(int& u, const int v) { (u -= v) < 0 && (u += MOD); }
inline void addeq(int& u, const int v) { (u += v) >= MOD && (u -= MOD); }
inline int sub(int u, const int v) { return (u -= v) < 0 ? u + MOD : u; }
inline int add(int u, const int v) { return (u += v) < MOD ? u : u - MOD; }
inline int mpow(int u, int v) {
int ret = 1;
for (; v; u = mul(u, u), v >>= 1) ret = mul(ret, v & 1 ? u : 1);
return ret;
} int main() {
freopen("water.in", "r", stdin);
freopen("water.out", "w", stdout); n = rint(), A = rint(), B = rint(), m = rint();
rep (i, 1, m) {
int s = rint(), t = rint(), a = rint();
link(s, t, a), ++ideg[t], ++odeg[s], addeq(osum[s], a), addeq(sum, a);
}
inv[1] = 1, sum = mpow(sum, MOD - 2);
rep (i, 2, m) inv[i] = mul(MOD - MOD / i, inv[MOD % i]); hd = 1;
rep (i, 1, A) que[++tl] = i, f[i] = g[i] = 1;
while (hd <= tl) {
int u = que[hd++], dlt = mul(f[u], mul(inv[odeg[u]],inv[odeg[u] - 1]));
addeq(g[u], mul(mul(osum[u], sum), mul(dlt, odeg[u])));
// extra input [cut(u,?)].
for (int i = head[u], v; i; i = graph[i].nxt) {
if (!--ideg[v = graph[i].to]) que[++tl] = v;
addeq(f[v], mul(f[u], inv[odeg[u]])); // normally flow.
addeq(g[v], mul(g[u], inv[odeg[u]])); // normally flow.
subeq(g[v], mul(mul(graph[i].old, sum), // extra output [cut(u,v)].
add(dlt, mul(f[u], inv[odeg[u]]))));
}
}
rep (i, n - B + 1, n) printf("%d%c", g[i], i < n ? ' ' : '\n');
return 0;
}

Solution -「多校联训」排水系统的更多相关文章

  1. Solution -「多校联训」I Love Random

    \(\mathcal{Description}\)   给定排列 \(\{p_n\}\),可以在其上进行若干次操作,每次选取 \([l,r]\),把其中所有元素变为原区间最小值,求能够得到的所有不同序 ...

  2. Solution -「多校联训」签到题

    \(\mathcal{Description}\)   Link.   给定二分图 \(G=(X\cup Y,E)\),求对于边的一个染色 \(f:E\rightarrow\{1,2,\dots,c\ ...

  3. Solution -「多校联训」朝鲜时蔬

    \(\mathcal{Description}\)   Link.   破案了,朝鲜时蔬 = 超现实树!(指写得像那什么一样的题面.   对于整数集 \(X\),定义其 好子集 为满足 \(Y\sub ...

  4. Solution -「多校联训」消失的运算符

    \(\mathcal{Description}\)   Link.   给定长度为 \(n\) 的合法表达式序列 \(s\),其中数字仅有一位正数,运算符仅有 - 作为占位.求将其中恰好 \(k\) ...

  5. Solution -「多校联训」假人

    \(\mathcal{Description}\)   Link.   一种物品有 长度 和 权值 两种属性,现给定 \(n\) 组物品,第 \(i\) 组有 \(k_i\) 个,分别为 \((1,a ...

  6. Solution -「多校联训」古老的序列问题

    \(\mathcal{Description}\)   Link.   给定序列 \(\{a_n\}\),和 \(q\) 次形如 \([L,R]\) 的询问,每次回答 \[\sum_{[l,r]\su ...

  7. Solution -「多校联训」Sample

    \(\mathcal{Description}\)   Link   (稍作简化:)对于变量 \(p_{1..n}\),满足 \(p_i\in[0,1],~\sum p_i=1\) 时,求 \(\ma ...

  8. Solution -「多校联训」光影交错

    \(\mathcal{Description}\)   Link.   一个游戏包含若干次卡牌抽取,每次以 \(p_l\) 的概率得到 \(+1\),\(p_d\) 的概率得到 \(-1\),否则得到 ...

  9. Solution -「多校联训」数学考试

    \(\mathcal{Description}\)   Link.   给定 \(n\) 个函数,第 \(i\) 个有 \(f_i(x)=a_ix^3+b_ix^2+cx_i+d~(x\in[l_i, ...

随机推荐

  1. PowerShell 【Switch篇】

    如果你学过其他的高级语言一定对Switch不陌生,下面讲解一下基本语法. 例1: 1 $n=Get-Random 5 2 $s='小明考试得分' 3 switch($n) 4 { 5 0 {$m=30 ...

  2. css 垂直居中技巧

    CSS垂直居中技巧,我只会23个,你会几个?自古以来(是有多?~),网页CSS的垂直居中需求始终没有停过,而其困难度也始终没有让人轻松过,经过了每位开发先烈的研究后,据说CSS的垂直居中技巧已达到近十 ...

  3. 【基因组学】maker的安装和注释

    本文默认读者有一定的生信基础,没有基础的可以阅读以前的笔记内容. maker作为比较受人认可的基因组注释软件,其流程较为清晰简单. 不知何故,我的conda无法安装maker,故而采用手动安装方式. ...

  4. CTF-sql-group by报错注入

    本文章主要涉及group by报错注入的原理讲解,如有错误,望指出.(附有目录,如需查看请点右下角) 一.下图为本次文章所使用到 user表,该表所在的数据库为 test 二.首先介绍一下本文章所使用 ...

  5. 《剑指offer》面试题45. 把数组排成最小的数

    问题描述 输入一个正整数数组,把数组里所有数字拼接起来排成一个数,打印能拼接出的所有数字中最小的一个.   示例 1: 输入: [10,2] 输出: "102" 示例 2: 输入: ...

  6. leetcode 33. 搜索旋转排序数组 及 81. 搜索旋转排序数组 II

    33. 搜索旋转排序数组 问题描述 假设按照升序排序的数组在预先未知的某个点上进行了旋转. ( 例如,数组 [0,1,2,4,5,6,7] 可能变为 [4,5,6,7,0,1,2] ). 搜索一个给定 ...

  7. 自从学会了VBA字典,VLOOKUP都不那么香了

    上篇博文中,小爬曾多次为VBA字典带货.鼓励多用字典,可以让我们的VBA脚本工具执行更快.今天小爬来细聊一下VBA字典的具体应用!如果你有一定VBA基础,那么看完你一定会对VBA字典有全新的认识:如果 ...

  8. [C# 学习]委托和线程

    委托有点像C语言的函数指针,简单总结一下如何使用委托. 1. 声明一个委托 public delegate void LabelSetEventHandler(Label la, string str ...

  9. 云计算实验二 Docker实验-mysql、tomcat、sqlserver、redis、oracle、mongo安装

    mysql安装 下载镜像 docker pull mysql 查看镜像 docker images|grep mysql 创建启动容器 docker run --name mysqlserver -v ...

  10. 根据happens-before法则借助同步

    在文章的开始,我们先来看一段代码以及他的执行情况: public class PossibleRecording{ static int x = 0, y = 0; static int a = 0, ...