Description

题库链接

有 \(n\) 个敌方单位,初始第 \(i\) 个单位的血量为 \(m_i\) 。共 \(Q\) 次操作,分两种:

  1. 对某一个单位以 \(p\) 的概率造成 \(1\) 点伤害。
  2. 对 \(k\) 个指定的敌方单位施放技能,只能命中恰好 \(1\) 个敌方单位。命中每个存活的敌方单位的概率是相等的(也就是说已经死亡的敌方单位不会有任何影响)。求命中各敌人的概率分别是多少。该类操作不会超过 \(C\) 次。

最后要求每个敌方单位的血量的期望。

\(n \leq 200 , Q \leq 200000 , C \leq 1000 , m_i \leq 100\)

Solution

不妨记 \(pk_{i,j}\) 表示第 \(i\) 个单位还剩血量为 \(j\) 时的概率。这样是可以用 \(O(Q\times m_i)\) 来维护的。

用 \(pk\) 数组我们可以计算出第 \(i\) 个单位存活概率为 \(p_i\) 。

得到这个概率,我们可以用 \(O(Cn^3)\) 的 \(\text{dp}\) (背包)来统计答案。

另 \(f_{i,j}\) 表示前 \(i\) 个人中存活了 \(j\) 个的概率。通过改变转移顺序第一维可以搞掉。

考虑优化,注意到可以逆推,及不用算每个人的概率时剩下的人重新求一次。大致是:

假设放第 \(i\) 个人时, \(f\) 数组是:

\[x_0,x_1,x_2,x_3,\cdots\]

那么第 \(i\) 个人转移后的状态为:

\[x_0',x_1',x_2',x_3',\cdots=x_0(1-p_i),x_0p_i+x_1(1-p_i),x_1p_i+x_2(1-p_i),x_2p_i+x_3(1-p_i),\cdots\]

显然逆推回去 \(x_0=\frac{x_0'}{1-p_i}\) ,剩下的满足:

\[x_i=\frac{x_i'-x_{i-1}p_i}{1-p_i}\]

那么就可以逆背包过程了,可以优化到 \(O(Cn^2)\) 。

Code

#include <bits/stdc++.h>
using namespace std;
const int N = 200+5, M = 100+5, yzh = 998244353; int n, m[N], pk[N][M], u, v, q, op, p, id, pa[N], k, lst[N], inv[N], f[N], invpa[N]; int quick_pow(int a, int b) {
int ans = 1;
while (b) {
if (b&1) ans = 1ll*ans*a%yzh;
a = 1ll*a*a%yzh, b >>= 1;
}
return ans;
}
int cal(int x) {
int ans = 0;
for (int i = 0; i < k; i++) (ans += 1ll*inv[i+1]*pa[x]%yzh*f[i]%yzh) %= yzh;
return ans;
}
void work() {
scanf("%d", &n);
for (int i = 1; i <= n; i++) scanf("%d", &m[i]), pk[i][m[i]] = 1, pa[i] = 1;
inv[0] = inv[1] = 1; for (int i = 2; i <= n; i++) inv[i] = -1ll*yzh/i*inv[yzh%i]%yzh;
scanf("%d", &q);
while (q--) {
scanf("%d", &op);
if (op == 0) {
scanf("%d%d%d", &id, &u, &v); p = 1ll*u*quick_pow(v, yzh-2)%yzh;
pk[id][0] = (pk[id][0]+1ll*pk[id][1]*p%yzh)%yzh;
for (int i = 1; i <= m[id]; i++)
pk[id][i] = (1ll*pk[id][i]*(1-p)%yzh+1ll*pk[id][i+1]*p%yzh)%yzh;
pa[id] = 1-pk[id][0]; invpa[id] = quick_pow(pk[id][0], yzh-2);
}else {
scanf("%d", &k); f[0] = 1;
for (int i = 1; i <= k; i++) scanf("%d", &lst[i]), f[i] = 0;
for (int i = 2; i <= k; i++) {
for (int j = i-1; j >= 1; j--)
f[j] = (1ll*(1-pa[lst[i]])*f[j]%yzh+1ll*pa[lst[i]]*f[j-1]%yzh)%yzh;
f[0] = 1ll*f[0]*(1-pa[lst[i]])%yzh;
}
printf("%d ", (cal(lst[1])+yzh)%yzh);
for (int i = 2; i <= k; i++) {
if ((pa[lst[i]]+yzh)%yzh != 1) {
f[0] = 1ll*f[0]*invpa[lst[i]]%yzh;
for (int j = 1; j < k; j++)
f[j] = 1ll*(1ll*f[j]-1ll*pa[lst[i]]*f[j-1]%yzh)%yzh*invpa[lst[i]]%yzh;
}else for (int i = 0; i < k; i++) f[i] = f[i+1];
for (int j = k-1; j >= 1; j--)
f[j] = (1ll*(1-pa[lst[i-1]])*f[j]%yzh+1ll*pa[lst[i-1]]*f[j-1]%yzh)%yzh;
f[0] = 1ll*f[0]*(1-pa[lst[i-1]])%yzh;
printf("%d ", (cal(lst[i])+yzh)%yzh);
}
puts("");
}
}
for (int i = 1; i <= n; i++) {
int ans = 0;
for (int j = 1; j <= m[i]; j++) (ans += 1ll*j*pk[i][j]%yzh) %= yzh;
printf("%d ", (ans+yzh)%yzh);
}
}
int main() {work(); return 0; }

[CTSC 2018]假面的更多相关文章

  1. CTSC 2018 游记

    day0 李总提前一天放假,回家颓整理行李... 然而我... 早上:睡觉... 中午:睡觉... 晚上:睡觉去火车站... 吃了几把鸡,本来想带李总入坑,但他挥手拒绝然后被李总带进了炸金花的坑... ...

  2. WC 2018/CTSC 2018/APIO 2018 游记

    (要写CTSC的时候才想起来没写WC2018,那就粗略回顾一下吧hhhhh) WC 2018(简略版): 大概和 一个宁夏和一个天津的大哥一个宿舍hhhh,字典序分宿舍真是奇妙. WC讲课真的不是人听 ...

  3. CTSC 2018酱油记

    Day0 5.5 花了一上午的时间把codechef div2的前四题切了,又在zbq老司机的指导下把第五题切了 中午12:00 gryz电竞组从机房出发,临走的时候看到很多学长挺恋恋不舍的,毕竟可能 ...

  4. [CTSC 2018] 混合果汁

    [题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=5343 [算法] 对于每组询问 , 首先二分答案 显然 , 最优策略为优先选择价格低的 ...

  5. Solution -「CTSC 2018」「洛谷 P4602」混合果汁

    \(\mathcal{Description}\)   Link.   \(n\) 种果汁,第 \(i\) 种美味度为 \(d_i\),每升价格 \(p_i\),一共 \(l_i\) 升.\(m\) ...

  6. APIO 2018 游记

    上接CTSC 2018 游记 day1 早上大概八九点起来洗了个澡跑到隔壁寝发现 tj 还在??? 原来昨天晚上听错名字了... 下午一起去 wfj 王府井玩,陪李总逛逛奢侈品店... 走了两三个小时 ...

  7. About me & 一些置顶的博文

    About me 一只历史上最弱的 \(\text{hnoier}\) ... 身在 \(\text{hn}\) 弱校,除了在四大名校夹缝中生存,还要受到同校 \(\text{Julao}\) 的鄙视 ...

  8. NOIWC 2019 冬眠记【游记】

    在我的blog查看:https://www.wjyyy.top/wc2019 Day -1 上火车了,but手机没电了. Day 0 中午1点左右到了广州东站.接站只有南站和机场有,于是坐了一个多小时 ...

  9. 2018 CTSC&APIO 游记

    CTSC 居然是CTSC先考,弄得我有些意外. 5.6 早上5:30乘坐高铁来到北京,差不多下午了吧,具体几点忘记了,然后来到宾馆   试机也没有去,就直接在宾馆颓废了. 5.7 考试的第一天,6:3 ...

随机推荐

  1. HTTP的瓶颈

    http协议,1.0或1.1版本,一个连接智能发送一个请求,通信都是基于请求/响应方式的,请求只能从客户端发起(半双工),发送冗长的首部. 在交互比较频繁的社交网站等,一直轮训带来的网络及服务器压力是 ...

  2. [ACM_水题] UVA 12502 Three Families [2人干3人的活后分钱,水]

      Three Families  Three families share a garden. They usually clean the garden together at the end o ...

  3. 数据帮助类(DataHelper)

    /// <summary> /// 是否为空... /// </summary> /// <param name="str">数据值</p ...

  4. NetCore入门篇:(七)Net Core项目使用Controller之二

    一.简介 1.说明Post,Get定义的区别. 2.说明如何路由定义. 二.Get.Post定义 1.api不定义访问方式时,同时支持get 和 post.如果定义某种方式,则仅支持某种方式.具体看代 ...

  5. BootStrap 4正式版发布(原文翻译)

    原文链接:http://blog.getbootstrap.com/2018/01/18/bootstrap-4/ 关于Bootstrap 什么是Bootstrap Bootstrap,来自 Twit ...

  6. 接口interface和抽象类型abstract

    一.接口 接口不能被实例化 接口只能包含方法声明 接口的成员包括方法.属性.索引器.事件 接口中不能包含常量.字段(域).构造函数.析构函数.静态成员 接口中的所有成员默认为public,因此接口中不 ...

  7. 如何利用Python绘制一个爱心

    刚学习Python几周,闲来无事,突然想尝试画一个爱心,步骤如下: 打开界面 打开Python shell界面,具体是Python语言的IDLE软件脚本. 2.建立脚本 单击左上角’File’,再单击 ...

  8. hdoj1045 Fire Net(二分图最大匹配)

    题意:给出一个图,其中有 . 和 X 两种,. 为通路,X表示墙,在其中放炸弹,然后炸弹不能穿过墙,问你最多在图中可以放多少个炸弹? 这个题建图有点复杂orz. 建图,首先把每一行中的可以放一个炸弹的 ...

  9. grafana 运行

    1,下载好项目,然后进入到目录 键入: ./bin/grafana-server web 运行 https://www.waitig.com/grafana-config-and-run.html 2 ...

  10. method swizzing

    原理 类的方法类别中,选择子的名称通过映射表找到应该调用的方法.如下所示:  OC 的运行时提供了几个方法可以操作这张表.可以向其中新增选择子,改变选择子的实现,或者交换选择子映射到的指针.  在 ...