题目传送门

题意:

有 \(n\) 个怪物,第 \(i\) 个怪物初始血量为 \(m_i\)。有 \(Q\) 次操作:

  • 0 x u v,有 \(p=\frac{u}{v}\) 的概率令 \(m_x\) 减 \(1\),另外 \(1-p\) 的概率啥也不干。如果 \(m_x=0\),那么这个操作无效。
  • 1 k x[1] x[2] ... x[k],在 \(x_1,x_2,\dots,x_k\) 这 \(k\) 个怪物中的血量不为 \(0\) 的怪物中,随机选择一个怪物。问对于 \(i=1,2,\dots,k\),怪物 \(x_i\) 被选中的概率是多少。

另外,所有操作结束后,输出 \(n\) 个数表示怪物 \(i\) 的血量的期望值。

\(n \leq 200\),\(q \leq 2 \times 10^5\),\(1\) 操作的次数 \(\leq 10^3\),\(m_i \leq 100\)。

虽说是为了刷期望的题而来的,但其实这题的关键点不在于期望。

我们令 \(dp_{i,j}\) 表示在当前时刻,怪物 \(i\) 的血量为 \(j\) 的概率。

对于 \(0\) 操作而言,你就暴力修改 \(dp_{x,j}\) 的值,直接一遍背包就行了,时间复杂度 \(\mathcal O(m)\)。

本题难就难在 \(1\) 操作怎样处理。

最 naive 的想法是,枚举每个 \(x_i\),显然 \(x_i\) 的血量是不能为 \(0\) 的。

然后在剩下 \(k-1\) 个怪物中跑一遍背包,\(f_{i,j}\) 表示在前 \(i\) 个怪物中有 \(j\) 个血量不为 \(0\) 的怪物的概率,背包进行转移。

答案为 \((1-dp_{x_t,0})\times (\sum_{i=0}^{k-1}f_{k,i}\times\frac{1}{i+1})\)。

式子很好理解,第一个括号表示 \(x_t\) 的血量不为 \(0\) 的概率,第二个括号枚举了血量不为 \(0\) 的怪物个数并用乘法原理计算出概率累加答案。

但这样是 \(n^3\) 的。

我的另一个 naive 的想法是 \(f_{i,j}\) 表示在前 \(i\) 个怪物中有 \(j\) 个血量不为 \(0\) 的怪物的概率,\(g_{i,j}\) 表示在后 \(i\) 个怪物中有 \(j\) 个血量不为 \(0\) 的怪物的概率。

这样计算 \(f,g\) 的时间复杂度降到了 \(n^2\),但统计答案还是 \(n^3\) 的。

这就要用到从背包中删除元素这一技巧。

\(f_{i,j}\) 的状态转移式子为 \(f_{i,j}=p\times f_{i-1,j-1}+(1-p)\times f_{i-1,j}\)

当删除一个元素的时候,我们其实是已知 \(f_{i,j}\) 和 \(p\) 还原出 \(f_{i-1,j}\)。

注意到 \(f_{i-1,i}=0\),也就是说 \(f_{i,i}=p\times f_{i-1,i-1}\)。

于是我们倒着循环,用 \(f_{i-1,j}=\dfrac{f_{i,j+1}-p\times f_{i-1,j+1}}{1-p}\) 求出 \(f_{i-1,j}\)

然后按照之前的公式累加答案即可。

\(\mathcal O(qm+cn^2)\)

注意预处理逆元,否则时间复杂度会多一个 \(\log\)(CSP-S 2020 的悲惨回忆)

#include <bits/stdc++.h>
using namespace std;
#define fi first
#define se second
#define fz(i,a,b) for(int i=a;i<=b;i++)
#define fd(i,a,b) for(int i=a;i>=b;i--)
#define ffe(it,v) for(__typeof(v.begin()) it=v.begin();it!=v.end();it++)
#define fill0(a) memset(a,0,sizeof(a))
#define fill1(a) memset(a,-1,sizeof(a))
#define fillbig(a) memset(a,63,sizeof(a))
#define pb push_back
#define mp make_pair
typedef pair<int,int> pii;
typedef long long ll;
bool chkmin(int &x,int y){return ((x>y)?(x=y,1):0);}
bool chkmax(int &x,int y){return ((x<y)?(x=y,1):0);}
const int MOD=998244353;
const int MAXN=200+5;
const int MAXM=100+5;
int n,m,x[MAXN];
ll dp[MAXN][MAXM],iv[MAXN],tmp[MAXN],f[MAXN][MAXN],inv[MAXN];
ll qpow(ll x,int e){
ll ret=1;
while(e){
if(e&1) ret=ret*x%MOD;
x=x*x%MOD;e>>=1;
} return ret;
}
int main(){
scanf("%d",&n);
for(int i=1;i<=n;i++){
int x;scanf("%d",&x);dp[i][x]=1;chkmax(m,x);iv[i]=1;
}
for(int i=1;i<=n;i++) inv[i]=qpow(i,MOD-2);
int Q;scanf("%d",&Q);while(Q--){
int opt;scanf("%d",&opt);
if(opt==0){
int x,u,v;scanf("%d%d%d",&x,&u,&v);
ll p=1ll*u*qpow(v,MOD-2)%MOD;
memset(tmp,0,sizeof(tmp));
for(int i=0;i<=m;i++){
if(i==0) tmp[i]=(tmp[i]+dp[x][i])%MOD;
else{
tmp[i]=(tmp[i]+dp[x][i]*(1-p+MOD)%MOD)%MOD;
tmp[i-1]=(tmp[i-1]+dp[x][i]*p%MOD)%MOD;
}
}
for(int i=0;i<=m;i++) dp[x][i]=tmp[i];
iv[x]=qpow(1-dp[x][0]+MOD,MOD-2);
} else {
int k;scanf("%d",&k);
for(int i=1;i<=k;i++) scanf("%d",&x[i]);
memset(f,0,sizeof(f));f[0][0]=1;
for(int i=1;i<=k;i++){
for(int j=0;j<=i;j++){
f[i][j]=f[i-1][j]*dp[x[i]][0]%MOD;
if(j) f[i][j]=(f[i][j]+f[i-1][j-1]*(1-dp[x[i]][0]+MOD)%MOD)%MOD;
}
}
// for(int i=0;i<=k;i++) printf("%lld ",f[k][i]);printf("\n");
for(int i=1;i<=k;i++){
ll ans=0;memset(tmp,0,sizeof(tmp));
for(int j=k-1;~j;j--)
tmp[j]=(f[k][j+1]-tmp[j+1]*dp[x[i]][0]%MOD+MOD)%MOD*iv[x[i]]%MOD;
// for(int j=0;j<=k;j++) printf("%lld ",tmp[j]);
// printf("\n");
for(int j=0;j<k;j++)
ans=(ans+(tmp[j]*(1-dp[x[i]][0]+MOD)%MOD)*inv[j+1]%MOD)%MOD;
printf("%lld ",ans);
} printf("\n");
}
}
for(int i=1;i<=n;i++){
ll ans=0;
for(int j=0;j<=m;j++) ans=(ans+1ll*j*dp[i][j]%MOD)%MOD;
printf("%lld ",ans);
} printf("\n");
return 0;
}

洛谷 P4564 [CTSC2018]假面(期望+dp)的更多相关文章

  1. [CTSC2018] 假面 | 期望 DP

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

  2. 洛谷P3600 随机数生成器(期望dp 组合数)

    题意 题目链接 Sol 一条重要的性质:如果某个区间覆盖了另一个区间,那么该区间是没有用的(不会对最大值做出贡献) 首先不难想到枚举最终的答案\(x\).这时我们需要计算的是最大值恰好为\(x\)的概 ...

  3. 洛谷P3600随机数生成器——期望+DP

    原题链接 写到一半发现写不下去了... 所以orz xyz32768,您去看这篇题解吧,思路很清晰,我之前写的胡言乱语与之差距不啻天渊 #include <algorithm> #incl ...

  4. 洛谷 P5279 - [ZJOI2019]麻将(dp 套 dp)

    洛谷题面传送门 一道 dp 套 dp 的 immortal tea 首先考虑如何判断一套牌是否已经胡牌了,考虑 \(dp\)​​​​​.我们考虑将所有牌按权值大小从大到小排成一列,那我们设 \(dp_ ...

  5. 洛谷2344 奶牛抗议(DP+BIT+离散化)

    洛谷2344 奶牛抗议 本题地址:http://www.luogu.org/problem/show?pid=2344 题目背景 Generic Cow Protests, 2011 Feb 题目描述 ...

  6. Lightning Conductor 洛谷P3515 决策单调性优化DP

    遇见的第一道决策单调性优化DP,虽然看了题解,但是新技能√,很开森. 先%FlashHu大佬,反正我是看了他的题解和精美的配图才明白的,%%%巨佬. 废话不多说,看题: 题目大意 已知一个长度为n的序 ...

  7. 洛谷P1541 乌龟棋(四维DP)

    To 洛谷.1541 乌龟棋 题目背景 小明过生日的时候,爸爸送给他一副乌龟棋当作礼物. 题目描述 乌龟棋的棋盘是一行N个格子,每个格子上一个分数(非负整数).棋盘第1格是唯一的起点,第N格是终点,游 ...

  8. 【洛谷】P1052 过河【DP+路径压缩】

    P1052 过河 题目描述 在河上有一座独木桥,一只青蛙想沿着独木桥从河的一侧跳到另一侧.在桥上有一些石子,青蛙很讨厌踩在这些石子上.由于桥的长度和青蛙一次跳过的距离都是正整数,我们可以把独木桥上青蛙 ...

  9. 【题解】洛谷P1052 [NOIP2005TG] 过河(DP+离散化)

    题目来源:洛谷P1052 思路 一开始觉得是贪心 但是仔细一想不对 是DP 再仔细一看数据不对 有点大 如果直接存下的话 显然会炸 那么就需要考虑离散化 因为一步最大跳10格 那么我们考虑从1到10都 ...

随机推荐

  1. [no code][scrum meeting] Beta 8

    $( "#cnblogs_post_body" ).catalog() 例会时间:5月22日15:30,主持者:赵涛 下次例会时间:5月23日11:30,主持者:肖思炀 一.工作汇 ...

  2. BUAA SE 个人项目作业

    项目 内容 这个作业属于哪个课程 2020春季计算机学院软件工程(罗杰 任健) 这个作业的要求在哪里 个人项目作业 我在这个课程的目标是 通过个人项目实践熟悉个人开发流程 一.在文章开头给出教学班级和 ...

  3. 认识spring security

    在一个系统中认证和授权是常有的事情,现在比较流行的框架有spring security.shiro等等.他们都能很好的帮助我们完成认证和授权的功能.那么假如说让我们自己完成一个登录那么应该大致的流程是 ...

  4. Netty:Netty中的零拷贝(Zero Copy)

    零复制概念: " 零复制"描述了计算机操作,其中CPU不执行将数据从一个存储区复制到另一个存储区的任务.通过网络传输文件时,通常用于节省CPU周期和内存带宽. WIKI的定义中,我 ...

  5. Flink计算pv和uv的通用方法

    PV(访问量):即Page View, 即页面浏览量或点击量,用户每次刷新即被计算一次. UV(独立访客):即Unique Visitor,访问您网站的一台电脑客户端为一个访客.00:00-24:00 ...

  6. 『学了就忘』Linux基础 — 15、了解Linux系统的目录结构

    目录 1.一级目录说明 (1)一级目录列表 (2)/bin/和/sbin/目录说明 (3)/boot/目录说明 (4)/lib/和/lib64/目录说明 (5)/lost+found/目录说明 (6) ...

  7. java性能优化常用工具jps、jstat、jinfo

    jps:虚拟机进程状况工具 jps可以用来查看虚拟机进程,基本等同于ps -ef|grep java #查看jps的使用文档 [root@localhost script]# jps -help us ...

  8. (四)php连接apache ,使用php-fpm方式

    上面各篇记录了编译安装lamp的各个部分,下面主要解决php和apache的连接问题.通过 php-fpm 连接. 连接前环境检查: php -v PHP 5.6.30 (cli) (built: O ...

  9. 关于docker中容器可以Ping通外网,真机无法Ping通容器的问题

    首先我们要知道整体的框架结构,docker是我们安装在centos7上的,而centos7是安装在vmware上.其中docker中还有若干容器运行. 整体框架图如下: 我们将它分为两部分,一部分是d ...

  10. 这些年我@yangbodong22011参与的开源

    2020年第一天,水一篇博客,对新年起码的尊重.这里记录下我参与的开源项目情况. Talk is cheap. Show me the code Linus Torvalds Jedis PR:htt ...