题面传送门

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

根据期望的线性公式,我们求出每个卡牌被发动的概率 \(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. 【数据结构与算法Python版学习笔记】基本数据结构——列表 List,链表实现

    无序表链表 定义 一种数据项按照相对位置存放的数据集 抽象数据类型无序列表 UnorderedList 方法 list() 创建一个新的空列表.它不需要参数,而返回一个空列表. add(item) 将 ...

  2. 剑指offer:JZ8 二叉树的下一个结点

    JZ8 二叉树的下一个结点 描述 给定一个二叉树其中的一个结点,请找出中序遍历顺序的下一个结点并且返回.注意,树中的结点不仅包含左右子结点,同时包含指向父结点的next指针.下图为一棵有9个节点的二叉 ...

  3. Beta阶段第二次会议

    时间:2020.5.18 工作进展 姓名 工作 难度 完成度 ltx 1.在开小程序开发文档,学习相关知识 轻 85% xyq 1.完成活动场地申请可视化代码(耗时半天) 中 100% lm 1.设计 ...

  4. BUAA SE 软件案例分析-CSDN

    Q A 这个作业属于哪个课程 2020春季计算机学院软件工程(罗杰 任健) 这个作业的要求在哪里 个人博客作业-软件案例分析 我在这个课程的目标是 系统地学习软件工程开发知识,掌握相关流程和技术,提升 ...

  5. 通过串口利用printf函数输出数据

    一.printf函数格式 printf函数具有强大的输出功能 %表示格式化字符串输出 目前printf支持以下格式的输出,例如: printf("%c",a);输出单个字符. pr ...

  6. Taylor公式原来可以这么简单

    1.Taylor公式 解决:含有高阶导数的中值定理或定积分.极限运算等题目 条件:f(x)在x=x0领域内(n+1)阶可导 结论:f(x)=Pn(x)+Rn(x) 2.x和x0的取值 3.Taylor ...

  7. Photoshop cc 绿色版 最新版 下载

    Photoshop cc 绿色版 下载 Photoshop cc 绿色版 最新版下载百度网盘下载 Photoshop 下载提取码: dh6z 作为一个程序员, 不懂点基本的作图都不配"新时代 ...

  8. copy-list-with-random-pointer leetcode C++

    A linked list is given such that each node contains an additional random pointer which could point t ...

  9. docker创建本地主机实例Virtualbox 驱动出错

    宿主机系统:Centos7 64位 创建主机实例Virtualbox 命令:docker-machine create -d virtualbox test 连接centos工具:Finalshell ...

  10. Linux&C———进程间通信

    管道和有名管道 消息队列 共享内存 信号 套接字 由于进程之间的并不会像线程那样共享地址空间和数据空间,所以进程之间就必须有自己特有的通信方式,这篇博客主要介绍自己了解到的几种进程之间的通信方式,内容 ...