题目描述

你正在玩你最喜欢的电子游戏,并且刚刚进入一个奖励关。在这个奖励关里,系统将依次随机抛出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. Java学习笔记五:Java中常用的运算符

    Java中常用的运算符 运算符是一种“功能”符号,用以通知 Java 进行相关的运算.譬如,我们需要将变量 score 的值设置为 20 ,这时候就需要一个“=”,告诉程序需要进行赋值操作. Java ...

  2. Python3爬虫(六) 解析库的使用之Beautiful Soup

    Infi-chu: http://www.cnblogs.com/Infi-chu/ Beautiful Soup 借助网页的结构和属性等特性来解析网页,这样就可以省去复杂的正则表达式的编写. Bea ...

  3. Python未彻底测试的项目

    第一 socket 第二 twisted 第三 tornado 第四 微信网页版本登录 第五:进程,线程,协程间关系 第六:TCP三次握手 第七:堡垒机 第八:重写django admin

  4. c++ combination by next_permutation

    #include <iostream> #include <algorithm> #include <vector> int main() { int n, r; ...

  5. 【廖雪峰老师python教程】day1

    主要内容摘要 函数参数[个人感觉难度很大,却很重要,可以先大概记一记]不要用的太复杂.戳这儿温习 递归函数:使用递归函数的优点是逻辑简单清晰,缺点是过深的调用会导致栈溢出.重点:找到递归关系和终止条件 ...

  6. CCF-NOIP-2018 提高组(复赛) 模拟试题(五)

    T1 相遇 [问题描述] 在一场奇怪的梦里,小 Y 来到了一个神奇的国度.这个国度可以用一根数轴表示,小 Y 在 N 处,而小 Y 想吃的美食在 K 处.小 Y 有两种方式移动, 一种叫做步行, 一种 ...

  7. SPOJ 694

    题面 题意: 给一个字符串,求它有多少个不同的子串 多组数据. Solution : 模板题,用所有的减去重复的即可. #include <cstdio> #include <alg ...

  8. 问题 C: 质因数的个数

    1947: 质因数的个数 时间限制: 1 Sec  内存限制: 32 MB提交: 245  解决: 114[提交][状态][讨论版][命题人:外部导入] 题目描述 求正整数N(N>1)的质因数的 ...

  9. apollo命令行传入参数

    Java apollo客户端运行配置 需要在META-INF中创建app.properties文件,以配置app.id 还需要在/opt/settings/server.properties或C:/o ...

  10. 树莓派初次使用的基本配置.md

    记录了一下树莓派初次使用的配置过程,包括装系统.修改 IP 等等. 树莓派(英语:Raspberry Pi),是一款基于 Linux 的单板机电脑. 它由英国的树莓派基金会所开发,目的是以低价硬件及自 ...