题目

先来考虑一下第一问,血量有\(P\)的概率减\(1\)

由于我们最后需要求每一个人的期望血量,于是考虑维护出每个人处于不同血量时候的概率

一个简单\(dp\)即可

\[dp_{i,j}=dp_{i,j+1}P+dp_{i,j}\times (1-P)
\]

\(dp_{i,j}\)表示第\(i\)个人血量为\(j\)的概率

第二问,发现概率的选中的人中的存活人数有关

于是每一个人只有两种状态,生或者是死,我们考虑求出每种存活人数对应的概率是多少

显然这里我们有一个这样的\(dp\)

\[f_{i,j}=f_{i-1,j-1}\times p_i+f_{i-1,j}\times (1-p_i)
\]

\(f_{i,j}\)表示前\(i\)个人里存活\(j\)个人的概率

发现做一次这个\(dp\)是\(O(n^2)\)的,对每一个人做一遍复杂度就是\(O(n^3)\)了

考虑消除某一个人的影响

设\(f_{j}=f_{n,j},g_{j}=g_{n-1,j}\),我们要消除第\(n\)的人的影响

显然边界条件有

\[f_0=g_0\times (1-p_n)
\]

于是

\[g_0=\frac{f_0}{1-p_n}
\]

求得了\(g_0\),我们就可以递推了

\[f_j=g_{j-1}\times p_n+g_{j}\times (1-p_n)
\]

于是就有

\[g_j=\frac{f_j-g_{j-1}\times p_n}{1-p_n}
\]

但是需要特别注意\(p_n=1\)的时候我们需要特殊处理一波,也非常简单就是

\[g_{j-1}=f_j
\]

代码

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#define re register
#define LL long long
#define max(a,b) ((a)>(b)?(a):(b))
#define min(a,b) ((a)<(b)?(a):(b))
const int maxn=205;
const LL mod=998244353;
inline int read() {
char c=getchar();int x=0;while(c<'0'||c>'9') c=getchar();
while(c>='0'&&c<='9') x=(x<<3)+(x<<1)+c-48,c=getchar();return x;
}
int n,h[maxn],m,opt,a[maxn];
LL dp[maxn][105],t[maxn],f[maxn],inv[maxn],g[maxn];
inline LL ksm(LL a,LL b) {
LL S=1;
while(b) {if(b&1) S=S*a%mod;b>>=1;a=a*a%mod;}
return S;
}
int main() {
n=read();
for(re int i=1;i<=n;i++) h[i]=read(),dp[i][h[i]]=1;
m=read();
inv[1]=1;
for(re int i=2;i<=n;i++) inv[i]=(mod-mod/i)*inv[mod%i]%mod;
while(m--) {
opt=read();
if(opt==0) {
int pos=read();
LL u=read(),v=read();
LL P=ksm(v,mod-2)*u%mod;
for(re int i=0;i<=h[pos];i++)
dp[pos][i]=(dp[pos][i]*(1-P+mod)%mod+dp[pos][i+1]*P%mod)%mod;
}
if(opt==1) {
int k=read();
for(re int i=1;i<=k;i++) a[i]=read();
for(re int i=1;i<=k;i++) {
t[a[i]]=0;
for(re int j=1;j<=h[a[i]];j++)
t[a[i]]=(t[a[i]]+dp[a[i]][j])%mod;
}
memset(f,0,sizeof(f));
f[0]=1;
for(re int i=1;i<=k;i++)
for(re int j=i;j>=0;--j) {
f[j]=(f[j]*(1-t[a[i]]+mod)%mod)%mod;
if(j-1>=0) f[j]=(f[j]+f[j-1]*t[a[i]]%mod)%mod;
}
for(re int i=1;i<=k;i++) {
memset(g,0,sizeof(g));
if(t[a[i]]!=1) {
LL d=ksm((1-t[a[i]]+mod)%mod,mod-2);
g[0]=f[0]*d%mod;
for(re int j=1;j<k;j++)
g[j]=(f[j]-g[j-1]*t[a[i]]%mod+mod)*d%mod;
}
else for(re int j=1;j<=k;j++) g[j-1]=f[j];
LL ans=0;
for(re int j=0;j<k;j++)
ans=(ans+g[j]*inv[j+1]%mod)%mod;
printf("%lld ",ans*t[a[i]]%mod);
}
putchar(10);
}
}
for(re int i=1;i<=n;i++) {
LL now=0;
for(re int j=1;j<=h[i];j++)
now=(now+dp[i][j]*(LL)j%mod)%mod;
printf("%lld ",now);
}
return 0;
}

[CTSC2018]假面的更多相关文章

  1. BZOJ5340: [Ctsc2018]假面

    BZOJ5340: [Ctsc2018]假面 https://lydsy.com/JudgeOnline/problem.php?id=5340 分析: 背包,只需要求\(g_{i,j}\)表示强制活 ...

  2. [CTSC2018] 假面 | 期望 DP

    题目链接 LOJ 2552 Luogu P4564 考场上这道题我先是写了个70分暴力,然后发现似乎可以NTT,然鹅问题是--我没学过NTT,遂脑补之,脑补出来了,下午出成绩一看,卡成暴力分(70)- ...

  3. BZOJ5340: [Ctsc2018]假面【概率+期望】【思维】

    LINK 思路 首先考虑减血,直接一个dp做过去,这个部分分不难拿 然后是\(op=1\)的部分 首先因为要知道每个人被打的概率,所以需要算出这个人活着的时候有多少个人活着时概率是什么 那么用\(g_ ...

  4. BZOJ5340 & 洛谷4564 & LOJ2552:[CTSC2018]假面——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=5340 https://www.luogu.org/problemnew/show/P4564 ht ...

  5. bzoj 5340: [Ctsc2018]假面

    Description 题面 Solution 生命值范围比较小,首先维护每一个人在每个血量的概率,从而算出生存的概率,设为 \(a[i]\) 询问时,只需要考虑生存的人数,可以 \(DP\) 设 \ ...

  6. BZOJ5340 [Ctsc2018]假面 【概率dp】

    题目链接 BZOJ5340 题解 我们能很容易维护每个人当前各种血量的概率 设\(p[u][i]\)表示\(u\)号人血量为\(i\)的概率 每次攻击的时候,讨论一下击中不击中即可转移 是\(O(Qm ...

  7. [CTSC2018]假面(概率DP)

    考场上以为CTSC的概率期望题都不可做,连暴力都没写直接爆零. 结果出来发现全场70以上,大部分AC,少于70的好像极少,感觉血亏. 设a[i][j]表示到当前为止第i个人的血量为j的概率(注意特判血 ...

  8. 并不对劲的bzoj5340:loj2552:uoj399:p4564: [Ctsc2018]假面

    题目大意 有\(n\)(\(n\leq200\))个非负整数\(m_1,m_2,...,m_n\)(\(\forall i\in[1,n],m_i\leq100\)),有\(q\)(\(q\leq2* ...

  9. UOJ399 CTSC2018 假面 期望、DP

    传送门 \(Q \leq 200000 , C \leq 1000 , m_i \leq 100\)-- 先考虑如何维护最后一次操作时所有人的血量期望.不难发现我们需要的复杂度是\(O(Qm_i)\) ...

随机推荐

  1. window常用的快捷键

    1.win+r打开运行命令 2.appwiz.cpl打开程序面板,进行程序的安装.卸载 输入win+r打开运行命令,输入appwiz.cpl 3.services.msc打开服务,一般用于启动或者关闭 ...

  2. nodejs应用离线安装部署、卸载

    公司写的文档,直接粘贴过来了: 本文档提供node应用一键安装.部署.卸载说明,包含对应脚本文件 默认版本:1.nodejs:v6.11.2.linux-x642.全局npm包:pm2.supervi ...

  3. 二、spark SQL交互scala操作示例

    一.安装spark spark SQL是spark的一个功能模块,所以我们事先要安装配置spark,参考: https://www.cnblogs.com/lay2017/p/10006935.htm ...

  4. centos文件/文件夹操作-检查磁盘、内存、cpu使用情况-vi操作命令

    Part1:CentOS文件/文件夹操作 1.新建文件夹 即创建目录 mkdir 文件名 新建一个名为test的文件夹在home下 vi source1 mkdir /home/test 注意:当创建 ...

  5. 集合框架以及Map(一)

    集合又称容器,编程思想中对其的定义为持有对象 我们在使用集合或者数组时得到最多的异常就是数组下表越界异常 Java.lang.ArrayIndexOutOfBoundsException这篇文章我们不 ...

  6. 为什么要学习 UML?

    UML 的首要价值是沟通和理解.好的图形可以帮助沟通设计思想,尤其是要回避许多细节时,图形也可以帮助你理解软件系统或业务流程.作为团队的成员,尝试弄清楚某些东西时,图形有助于理解和沟通整个团队所理解到 ...

  7. Lucene原理之概念

    概念: 数据分两种: 1.结构化数据:指具有固定格式或有限长度的数据,如数据库,元数据等. 2.非结构化数据:指不定长或无固定格式的数据,如邮件,word文档等.(半结构化数据:如XML,HTML等, ...

  8. Spring入门(四)— 整合Struts和Hibernate

    一.Spring整合Struts 1. 初步整合 只要在项目里面体现spring和 strut即可,不做任何的优化. struts 环境搭建 创建action public class UserAct ...

  9. [转]Tips——Chrome DevTools - 25 Tips and Tricks

    Chrome DevTools - 25 Tips and Tricks 原文地址:https://www.keycdn.com/blog/chrome-devtools 如何打开? 1.从浏览器菜单 ...

  10. ES6学习笔记(三)-正则扩展

    PS: 前段时间转入有道云笔记,体验非常友好,所以笔记一般记录于云笔记中,每隔一段时间,会整理一下, 发在博客上与大家一起分享,交流和学习. 以下: