题面传送门

感觉是道挺好的题,可惜当时没写题解来着的?

根据期望的线性公式,我们求出每个卡牌被发动的概率 \(q_i\),然后

\[ans=\sum\limits_{i=1}^np_id_i
\]

于是我们求出 \(q_i\) 即可。

我们设 \(dp_{i,j}\) 表示在前 \(i\) 张牌里发动了 \(j\) 张牌的概率。

如果已知 \(dp_{i,j}\),那么可以这样求出 \(q_i\):

\[q_i=\sum\limits_{j=0}^rdp_{i-1,j}+(1-(1-p_i)^{r-j})
\]

稍微解释一下这个式子。我们枚举在前 \(i-1\) 个牌里面发动了 \(j\) 个牌。这意味着有 \(j\) 轮不会考虑到第 \(i\) 张牌。在剩下 \(r-j\) 轮当中,\(i\) 号卡牌一次都没被发动的概率为 \((1-p_i)^{r-j}\),\(1-(1-p_i)^{r-j}\) 就是卡牌 \(i\) 至少被发动一次的概率。

那么怎样求 \(dp_{i,j}\) 呢,其实用背包的套路就可以了。分两种情况:

  1. 如果卡牌 \(j\) 被选,那么 \(dp_{i,j}\) 可以从 \(dp_{i-1,j-1}\) 转移过来。有 \(r-j+1\) 轮会考虑到卡牌 \(i\),卡牌 \(i\) 发动的概率为 \((1-(1-p_i)^{r-j+1})\)。
  2. 如果卡牌 \(j\) 没被选,那么 \(dp_{i,j}\) 可以从 \(dp_{i-1,j}\) 转移过来。有 \(r-j\) 轮会考虑到卡牌 \(i\),卡牌 \(i\) 未被发动的概率为 \((1-p_i)^{r-j}\)。

综上 \(dp_{i,j}=dp_{i-1,j-1}\times(1-(1-p_i)^{r-j+1})+dp_{i,j}\times(1-p_i)^{r-j}\)

于是这题就做完了,复杂度 \(\mathcal O(Tnr)\)。

另外预处理 \(1-p_i\) 的幂。快速幂会多一个 \(\log\)

#include <bits/stdc++.h>
using namespace std;
#define fi first
#define se second
#define pb push_back
#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 foreach(it,v) for(__typeof(v.begin()) it=v.begin();it!=v.end();it++)
#define all(a) a.begin(),a.end()
#define fill0(a) memset(a,0,sizeof(a))
#define fill1(a) memset(a,-1,sizeof(a))
#define fillbig(a) memset(a,0x3f,sizeof(a))
#define y1 y10101010101
#define y0 y01010101010
typedef pair<int,int> pii;
typedef long long ll;
int T,n,r,d[222];
double p[222],dp[222][222];
double pw[222][222];
inline void clear(){
for(int i=1;i<=n;i++) d[i]=0;
for(int i=1;i<=n;i++) p[i]=0;
for(int i=0;i<=n;i++) for(int j=0;j<=n;j++) dp[i][j]=0;
for(int i=0;i<=n;i++) for(int j=0;j<=r;j++) pw[i][j]=0;
}
inline void solve(){
scanf("%d%d",&n,&r);clear();//remember to clear the data!
for(int i=1;i<=n;i++) scanf("%lf%d",&p[i],&d[i]);
for(int i=1;i<=n;i++){
pw[i][0]=1;
for(int j=1;j<=r;j++){
pw[i][j]=pw[i][j-1]*(1.0-p[i]);//calculate the power of 1-p[i]
// printf("%d %d %.10lf\n",i,j,pw[i][j]);
}
}
dp[0][0]=1;
for(int i=1;i<=n;i++) for(int j=0;j<=i;j++){
dp[i][j]=dp[i-1][j]*pw[i][r-j];
if(j) dp[i][j]+=dp[i-1][j-1]*(1-pw[i][r-j+1]);//calculate dp[i][j] using the fomula above
// printf("%d %d %.10lf\n",i,j,dp[i][j]);
}
double ans=0;
for(int i=1;i<=n;i++){
double prob=0;
for(int j=0;j<=i-1;j++){
prob+=dp[i-1][j]*(1-pw[i][r-j]);//calculate p[i]
}
// printf("%d %.10lf\n",i,prob);
ans+=prob*d[i];
}
printf("%.10lf\n",ans);
}
int main(){
scanf("%d",&T);
while(T--) solve();
return 0;
}
/*
2
3 2
0.5000 2
0.3000 3
0.9000 1
3 2
0.5000 2
0.3000 3
0.9000 1
*/

洛谷 P3239 [HNOI2015]亚瑟王(期望+dp)的更多相关文章

  1. 洛谷 P3239 [HNOI2015]亚瑟王(期望dp)

    题面 luogu 题解 一道复杂的期望\(dp\) 思路来源:__stdcall 容易想到,只要把每张牌打出的概率算出来就可以求出\(ans\) 设\(fp[i]\)表示把第\(i\)张牌打出来的概率 ...

  2. [洛谷 P3239] [HNOI2015]亚瑟王

    [HNOI2015]亚瑟王 题目描述 小 K 不慎被 LL 邪教洗脑了,洗脑程度深到他甚至想要从亚瑟王邪教中脱坑.他决定,在脱坑之前,最后再来打一盘亚瑟王.既然是最后一战,就一定要打得漂亮.众所周知, ...

  3. P3239 [HNOI2015]亚瑟王 期望dp

    这个题一看就是期望dp,但是我有个问题,一个事件的期望等于他所有事件可能行乘权值的和吗...为什么我有天考试的时候就不对呢...求大佬解释一下. 至于这道题,f[i][j]代表前i个有j个发动技能,这 ...

  4. 洛谷P3239 [HNOI2015]亚瑟王

    题目描述 小 K 不慎被 LL 邪教洗脑了,洗脑程度深到他甚至想要从亚瑟王邪教中脱坑.他决定,在脱坑之前,最后再来打一盘亚瑟王.既然是最后一战,就一定要打得漂亮.众所周知,亚瑟王是一个看脸的游戏,技能 ...

  5. P3239 [HNOI2015]亚瑟王 期望 dp

    LINK:亚瑟王 Saber!Excalibur! 比较难的期望dp. 可以发现如果暴力枚举所有的局面复杂度很高 . 转换的思路则是 期望的线性性. 求出每张牌的期望累加即可. 考虑每张牌的期望=这张 ...

  6. 洛谷P3239 [HNOI2015]亚瑟王(期望dp)

    传送门 stdcall大佬好强 期望的姿势不是很高……据大佬说期望有一个线性性质,也就是说可以把每一张牌的期望伤害算出来然后再加起来就是总的期望伤害 因为每一张牌只能用一次,我们设$dp[i]$表示第 ...

  7. P3239 [HNOI2015]亚瑟王——概率DP

    题面:亚瑟王 最近考试考期望很自闭啊,没做过这种类型的题,只能现在练一练: 所谓期望,就是状态乘上自己的概率:对于这道题来说,我们要求的是每张牌的伤害乘上打出的概率的和: 当然不是直接乘,因为给的是每 ...

  8. BZOJ4008: [HNOI2015]亚瑟王(期望dp)

    Time Limit: 20 Sec  Memory Limit: 512 MBSec  Special JudgeSubmit: 1952  Solved: 1159[Submit][Status] ...

  9. [HNOI2015]亚瑟王(期望+DP)

    题解 利用期望的线性性,可以把问题转化为求每一个卡牌造成期望的期望值. 然后我们就需要知道每一个卡牌发动技能的概率. 因为当某一张卡牌发动技能时这一轮会结束,这就很难直接计算了. 我们使用DP 设dp ...

随机推荐

  1. MyBatis的框架设计

    1.MyBatis的框架设计 2.整体设计 2.1 总体流程 (1)加载配置并初始化       触发条件:加载配置文件 配置来源于两个地方,一处是配置文件,一处是Java代码的注解,将SQL的配置信 ...

  2. UnboundLocalError: local variable 'range' referenced before assignment

    1. 报错信息 UnboundLocalError: local variable 'range' referenced before assignment 2. 代码 class Car(): &q ...

  3. go-zero 实战之 blog 系统

    go-zero 实战项目:blog 本文以 blog 的网站后台为例,着重介绍一下如何使用 go-zero 开发 blog 的用户模块. 本文涉及的所有资料都已上传 github 仓库 kougazh ...

  4. [软工顶级理解组] 团队任务拆解(Alpha)

    一.任务概述 在alpha阶段,我们需要完成功能规格说明书中所提到的所有功能,在一个阶段的开发周期内,交付最小可行的可用版本. 二.任务分配及时长 分组&成员 具体任务 预计时长(小时) 前端 ...

  5. 【学习笔记】Vizing 定理

    图染色问题的经典结论 定义 称一个边染色方案合法当且仅当每个顶点连出的所有边的颜色都互不相同,如果此时出现了 \(k\) 个颜色那么称该方案是图的一组 \(k\) 染色 一张无向图的边着色数为最小的 ...

  6. [CSP-S 2021] 回文

    题目描述: 给定正整数 n 和整数序列 a1, a2,-,a2n,在这 2n 个数中,1, 2,-,n 分别各出现恰好 2 次.现在进行 2n 次操作,目标是创建一个长度同样为 2n 的序列 b 1, ...

  7. Java并发:重入锁 ReentrantLock(一)

    ReentrantLock 是一种可重入的互斥锁,它不像 synchronized关键字一样支持隐式的重进入,但能够使一个线程(不同的方法)重复对资源的重复加锁而不受阻塞. ReentrantLock ...

  8. hdu 5083 Instruction (稍比较复杂的模拟题)

    题意: 二进制指令转汇编指令,汇编指令转二进制指令. 思路: 额,条理分好,想全,思维不能乱. 代码: int findyu(char yu[50],char c){ int l=strlen(yu) ...

  9. hdu 4771 Stealing Harry Potter's Precious (BFS+状压)

    题意: n*m的迷宫,有一些格能走("."),有一些格不能走("#").起始点为"@". 有K个物体.(K<=4),每个物体都是放在& ...

  10. 修改记事本PE结构弹计算器Shellcode

    目录 修改记事本PE结构弹计算器Shellcode 0x00 前言 0x01 添加新节 修改节数量 节表位置 添加新节表信息 0x02 添加弹计算器Shellcode 修改代码 0x03 修改入口点 ...