题目描述

你正在玩你最喜欢的电子游戏,并且刚刚进入一个奖励关。在这个奖励关里,系统将依次随机抛出k次宝物,每次你都可以选择吃或者不吃(必须在抛出下一个宝物之前做出选择,且现在决定不吃的宝物以后也不能再吃)。

宝物一共有n种,系统每次抛出这n种宝物的概率都相同且相互独立。也就是说,即使前k-1 次系统都抛出宝物1(这种情况是有可能出现的,尽管概率非常小),第k次抛出各个宝物的概率依然均为1/n。

获取第 i 种宝物将得到Pi分,但并不是每种宝物都是可以随意获取的。第i种宝物有一个前提宝物集合Si。只有当Si中所有宝物都至少吃过一次,才能吃第i 种宝物(如果系统抛出了一个目前不能吃的宝物,相当于白白的损失了一次机会)。注意,Pi 可以是负数,但如果它是很多高分宝物的前提,损失短期利益而吃掉这个负分宝物将获得更大的长期利益。

假设你采取最优策略,平均情况你一共能在奖励关得到多少分值?

输入输出格式

输入格式:

第一行为两个正整数k 和n,即宝物的数量和种类。以下n行分别描述一种

宝物,其中第一个整数代表分值,随后的整数依次代表该宝物的各个前提宝物(各

宝物编号为1到n),以0结尾。

输出格式:

输出一个实数,保留六位小数,即在最优策略下平均情况的得分。

输入输出样例


输入样例#1:

1 2
1 0
2 0

输出样例#1:

1.500000


输入样例#2:

6 6
12 2 3 4 5 0
15 5 0
-2 2 4 5 0
-11 2 5 0
5 0
1 2 4 5 0

输出样例#2:

10.023470


说明

1 <= k <= 100, 1 <= n <= 15,分值为[-106,106]内的整数。


题解

这是一道状压dp题,数据范围很小,只有15(很标准啊)

首先,解释一下题意,会有k个宝物掉下,共n种,所以每次每种宝物掉下的概率都是1/n,而题目最后说的最优策略是指这次掉下的宝物,你可以不选,这是因为它的贡献是负数且它对后面的宝物是没用的,平均情况是指每次掉下每种宝物的情况都是1/n,所以我们要将所得的期望得分/n,即

本轮期望=(上一轮期望+本轮得分)/n

而正向推的话可能会出现从合法情况推到不合法的情况,那么这种情况乱再推也是没用的,所以我们倒着推,保证统计结果时一定合法(听说最优策略的期望dp都是倒着推???),那么结果最后就保存在dp[1][0]

设dp[i][j]表示第i轮已经收集的宝物集合j的期望

那么状态转移方程就变成了这样

if(本宝物可以收集)

  dp[i][j]+=max(dp[i+1][j],dp[i+1][j|1<<(k-1)]+v[k])/n//v[]表示宝物价值

else

  dp[i][j]+=dp[i+1][j]/n;

 #include<bits/stdc++.h>
#define in(i) (i=read())
using namespace std;
int read()
{
int ans=,f=;
char i=getchar();
while(i<'' || i>'')
{
if(i=='-') f=-;
i=getchar();
}
while(i>='' && i<='')
{
ans=(ans<<)+(ans<<)+i-'';
i=getchar();
}
return ans*f;
}
int m,n;
int cur[];
int v[];
double dp[][];
int main()
{
in(m);in(n);
for(int i=;i<=n;i++)
{
in(v[i]);
int u;
in(u);
while(u)
{
cur[i]|=<<(u-);
in(u);
}
}
int tot=<<n;
for(int i=m;i>=;i--)
{
for(int j=;j<tot;j++)
{
for(int k=;k<=n;k++)
{
if((cur[k]&j)==cur[k]) dp[i][j]+=max(dp[i+][j],dp[i+][j|<<(k-)]+v[k])/n;
else dp[i][j]+=dp[i+][j]/n;
}
// dp[i][j]/=n;
}
}
printf("%0.6lf\n",dp[][]);
return ;
}
 #include<bits/stdc++.h>
#define in(i) (i=read())
using namespace std;
int read()
{
int ans=,f=;
char i=getchar();
while(i<'' || i>'')
{
if(i=='-') f=-;
i=getchar();
}
while(i>='' && i<='')
{
ans=(ans<<)+(ans<<)+i-'';
i=getchar();
}
return ans*f;
}
int m,n;
int cur[];
int v[];
double dp[][];
int main()
{
in(m);in(n);
for(int i=;i<=n;i++)
{
in(v[i]);
int u;
in(u);
while(u)
{
cur[i]|=<<(u-);
in(u);
}
}
int tot=<<n;
for(int i=m;i>=;i--)
{
for(int j=;j<tot;j++)
{
for(int k=;k<=n;k++)
{
if((cur[k]&j)==cur[k]) dp[i][j]+=max(dp[i+][j],dp[i+][j|<<(k-)]+v[k])/n;
else dp[i][j]+=dp[i+][j]/n;
}
// dp[i][j]/=n;
}
}
printf("%0.6lf\n",dp[][]);
return ;

SCOI2008奖励关 [状压dp]的更多相关文章

  1. 【BZOJ1076】[SCOI2008]奖励关 状压DP+期望

    [BZOJ1076][SCOI2008]奖励关 Description 你正在玩你最喜欢的电子游戏,并且刚刚进入一个奖励关.在这个奖励关里,系统将依次随机抛出k次宝物,每次你都可以选择吃或者不吃(必须 ...

  2. B1076 [SCOI2008]奖励关 状压dp&&期望dp

    这个题的n<15,一看就是状压dp.但是状态不是很好想.f[][]存i关的状态j. 这个题另一个关键思想在于倒推,我一开始想的是正推,但是只能记忆化了. 题干: 题目描述 你正在玩你最喜欢的电子 ...

  3. [BZOJ1076][SCOI2008]奖励关 状压dp

    1076: [SCOI2008]奖励关 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 3070  Solved: 1595[Submit][Statu ...

  4. BZOJ1076:[SCOI2008]奖励关(状压DP,期望)

    Description 你正在玩你最喜欢的电子游戏,并且刚刚进入一个奖励关.在这个奖励关里,系统将依次随机抛出k次宝物, 每次你都可以选择吃或者不吃(必须在抛出下一个宝物之前做出选择,且现在决定不吃的 ...

  5. 洛谷 P2473 [SCOI2008]奖励关(状压dp+期望)

    题面 luogu 题解 \(n \leq 15\) 状压 \(f[i][S]\)表示第\(i\)轮,吃过的集合为\(S\) 正着转移好像有点复杂 考虑逆推转移(正着转移应该也行) \(f[i][S]\ ...

  6. 洛谷P2473奖励关——状压DP

    题目:https://www.luogu.org/problemnew/show/P2473 还是对DP套路不熟悉... 像这种前面影响后面,而后面不影响前面的问题就应该考虑倒序递推: 看n只有15那 ...

  7. [SCOI2008]奖励关 - 状压动规 - 概率与期望

    Description 你正在玩你最喜欢的电子游戏,并且刚刚进入一个奖励关.在这个奖励关里,系统将依次随机抛出k次宝物,每次你都可以选择吃或者不吃(必须在抛出下一个宝物之前做出选择,且现在决定不吃的宝 ...

  8. 【洛谷】2473:[SCOI2008]奖励关【期望DP(倒推)】

    P2473 [SCOI2008]奖励关 题目背景 08四川NOI省选 题目描述 你正在玩你最喜欢的电子游戏,并且刚刚进入一个奖励关.在这个奖励关里,系统将依次随机抛出k次宝物,每次你都可以选择吃或者不 ...

  9. [SCOI2008]奖励关(期望dp)

    你正在玩你最喜欢的电子游戏,并且刚刚进入一个奖励关.在这个奖励关里,系统将依次随机抛出k次宝物,每次你都可以选择吃或者不吃(必须在抛出下一个宝物之前做出选择,且现在决定不吃的宝物以后也不能再吃). 宝 ...

随机推荐

  1. python3 练习题100例 (二十六)回文数判断

    题目内容: 给一个5位数,判断它是不是回文数,是则输出yes,不是则输出no. 例如12321是回文数,它的个位与万位相同,十位与千位相同. 输入格式: 共一行,为一个5位数. 输出格式: 共一行,y ...

  2. print(__file__)返回<encoding error>的问题

    今天写了一下代码,本来是想得到当前文件的上面三层的目录的,结果返回的却是错误 import os import sys print(__file__) # 得到上上层目录的路径之后,加入到默认的环境变 ...

  3. 《PHP内核探索系列文章》系列分享专栏

    <PHP内核探索系列文章>已整理成PDF文档,点击可直接下载至本地查阅 简介 PHP内核探索系列文章收藏夹收藏有关PHP内核方面的知识的文章,对PHP高级进阶的朋友提供PHP内核方面的知识 ...

  4. idea 普通文件夹 转换成 module

    经常会遇到从GitHub上download的progect在idea里面打开是普通文件夹形式,而并不是我们想要的module形式(文件夹图标右下角有个蓝色的tag),那么如何快速转换成我们想要的mod ...

  5. maven 添加自己下载的jar包到本地仓库

    1.在pom文件中添加依赖,其中groupId等变量都自拟. 例如: 2.在命令行执行以下命令,提示build success即表示安装成功. mvn install:install-file -Dg ...

  6. PATA1034题解

    题目:https://pintia.cn/problem-sets/994805342720868352/problems/994805456881434624 参考:算法笔记(胡凡)10.3.1 # ...

  7. LeetCode算法1—— 两数之和

    给定一个整数数组和一个目标值,找出数组中和为目标值的两个数. 你可以假设每个输入只对应一种答案,且同样的元素不能被重复利用. 示例: 给定 nums = [2, 7, 11, 15], target ...

  8. Create Fiori List App Report with ABAP CDS view – PART 2

    In the Part 1 blog, we have discussed below topics CDS annotations for Fiori List Report. How to cre ...

  9. WPF中的命令与命令绑定(二)

    原文:WPF中的命令与命令绑定(二) WPF中的命令与命令绑定(二)                                              周银辉在WPF中,命令(Commandi ...

  10. 通过py2exe打包python程序的过程中,解决的一系列问题

    py2exe的使用方法参考<py2exe使用方法>. 注:程序可以在解释器中正常运行,一切问题都出在打包过程中. 问题1: 现象:RuntimeError: maximum recursi ...