HDOJ.2955 Robberies (01背包+概率问题)
Robberies
算法学习—–动态规划初探
题意分析
有一个小偷去抢劫银行,给出来银行的个数n,和一个概率p为能够逃跑的临界概率,接下来有n行分别是这个银行所有拥有的钱数mi和抢劫后被抓的概率pi,求在不被抓的情况下,小偷能抢到的最多的钱是多少。
显然这是一道概率问题,计算小偷不能逃的概率是不好算的,不如计算他成功的概率。若把题目中每个数据变成能够逃跑的概率,那就是1-pi。
我们先举个简单的例子.
不妨假设有3个银行:
①如果小偷都能抢劫,那么抢劫后能逃跑的概率就是(1-p1) * (1-p2) * (1-p3),对应抢到的金钱就是m1+m2+m3
②若他只能抢劫其中2个,那么就是有下面3种情况
| ~ | case1 | case 2 | case3 |
|---|---|---|---|
| 成功概率 | (1-p1) * (1-p2) | (1-p1) * (1-p3) | (1-p2) * (1-p3) |
| 获得金钱 | m1+m2 | m1+m3 | m2+m3 |
③若他只能抢劫1个,那么他只有3中选择
④或者他一个都抢劫不了,那……
好了到这里就差不多理解题目的大意了。因为在概率里面,2件独立事情一起发生的概率是每个事件发生概率的乘积。(如果表述有不合理的地方,请各位留言指正)
我们接着看小偷,小偷面对每个银行,有偷或者不偷2种选择,这不就是典型的0/1背包问题的情景吗?那么限制小偷的偷东西的是什么呢?显然是他被抓的概率,同时小偷想获得最大价值的金钱,不妨我们做个对比:
背包容量(限制拿东西)-> 被抓概率(限制偷东西)
商品价值(希望最大) -> 金钱数量(希望最大)
到这里,大概就能看出来了,dp数组应该保存的是概率(开double类型),数组的下标代表所偷到的金钱。
下面仍有几点需要注意的(个人意见,仅供参考):
1.dp[0]要初始化为1,也很容易理解,啥都没有偷,被抓的概率为0,成功概率为100%;
2.dp完之后,应该由大到小遍历一遍,一旦有一种情况的成功概率大于等于(1-p) p为每组数据一开始给出来的那个临界,输出对应的金钱,break即可。
代码总览
/*
Title:HDOJ.2955
Auhtor:pengwill
Date:2016-12-18
*/
#include <iostream>
#include <algorithm>
#include <cstdio>
#include <cstring>
#include <cstdlib>
#define ma 105
using namespace std;
int m[ma];
double dp[10000],p[ma];
int main()
{
//freopen("in.txt","r",stdin);
int t;
scanf("%d",&t);
while(t--){
int n,i,j,sum = 0;double pmax;
scanf("%lf %d",&pmax,&n);
pmax = 1-pmax;
memset(m,0,sizeof(m));
memset(p,0,sizeof(p));
for(i = 1;i<=n;++i){
scanf("%d %lf",&m[i],&p[i]);
sum+=m[i];
p[i] = 1-p[i];
}
memset(dp,0,sizeof(dp));
dp[0] =1;
for(j = 1;j<=n;++j)
for(i = sum;i>=m[j];--i)
dp[i] = max(dp[i],dp[i-m[j]]*p[j]);
for(i = sum;i>=0;--i)
if(dp[i]>=pmax)
break;
printf("%d\n",i);
}
return 0;
}
HDOJ.2955 Robberies (01背包+概率问题)的更多相关文章
- HDOJ 2955 Robberies (01背包)
10397780 2014-03-26 00:13:51 Accepted 2955 46MS 480K 676 B C++ 泽泽 http://acm.hdu.edu.cn/showproblem. ...
- hdu 2955 Robberies 0-1背包/概率初始化
/*Robberies Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total S ...
- hdu 2955 Robberies (01背包)
链接:http://acm.hdu.edu.cn/showproblem.php?pid=2955 思路:一开始看急了,以为概率是直接相加的,wa了无数发,这道题目给的是被抓的概率,我们应该先求出总的 ...
- HDU 2955 Robberies(01背包变形)
Robberies Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total S ...
- hdu 2955 Robberies (01背包好题)
Robberies Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total S ...
- HDU——2955 Robberies (0-1背包)
题意:有N个银行,每抢一个银行,可以获得\(v_i\)的前,但是会有\(p_i\)的概率被抓.现在要把被抓概率控制在\(P\)之下,求最多能抢到多少钱. 分析:0-1背包的变形,把重量变成了概率,因为 ...
- HDU 2955 Robberies --01背包变形
这题有些巧妙,看了别人的题解才知道做的. 因为按常规思路的话,背包容量为浮点数,,不好存储,且不能直接相加,所以换一种思路,将背包容量与价值互换,即令各银行总值为背包容量,逃跑概率(1-P)为价值,即 ...
- HDU 2955 Robberies(01背包)
Robberies Problem Description The aspiring Roy the Robber has seen a lot of American movies, and kno ...
- Robberies(HDU2955):01背包+概率转换问题(思维转换)
Robberies HDU2955 因为题目涉及求浮点数的计算:则不能从正面使用01背包求解... 为了能够使用01背包!从唯一的整数(抢到的钱下手)... 之后就是概率的问题: 题目只是给出被抓的 ...
随机推荐
- Bootstrap基础篇—常见的CSS类
一.标题 标签 大小 h1 36px h2 30px h3 24px h4 18px h5 14px h6 12px 二.常见的内联样式 标签 用途 del 删除的文本 s 无用的文本 ins 插入的 ...
- 提权基础-----mysql-udf提权
1.总结关于udf提权方法 通过弱口令,爆破,网站配置文件等方式得到mysql数据库帐号密码,---还要能外连 (1).将udf.dll代码的16进制数声明给my_udf_a变量 set @my_ud ...
- keepalived+nginx实现高可用+tomcat
1.keepalived的yum安装 安装依赖包[root@localhost ~]# yum install -y curl gcc openssl-devel libnl3-devel net-s ...
- Linux命令应用大词典-第28章 硬件管理
28.1 lscpu:显示有关CPU架构的信息 28.2 nproc:显示当前进程可用的CPU数目 28.3 chcpu:配置CPU
- 【outPut_Class 输出类】使用说明
对象:outPut 说明:定义输出结果类的相关操作.此对象的核心是[JSON]类,所以它继承了[JSON]类的所有方法 重要: 输出结果样式为XML时,自带根节点"root".输出 ...
- JavaScript 字符串 & Math & Date
字符串 字符串就是零个或多个排在一起的字符,放在单引号或双引号之中. 'abc' "abc" 单引号字符串的内部,可以使用双引号.双引号字符串的内部,可以使用单引号. 'key=& ...
- 【机器学习】多项式回归sklearn实现
[机器学习]多项式回归原理介绍 [机器学习]多项式回归python实现 [机器学习]多项式回归sklearn实现 使用sklearn框架实现多项式回归.使用框架更方便,可以少写很多代码. 使用一个简单 ...
- 安装HIVE
参考:https://cwiki.apache.org/confluence/display/Hive/GettingStarted 1.下载hive安装包 到apache官网或者其它地方下载 ...
- hadoop参数(未完).md
我X,有违禁词.麻烦提醒一下哪个词好吗?
- Windows Phone编程回顾
前言 已有一年多没有碰WP相关的开发了. 近期经常看博客园的文章, 发现开发WP应用的同学很多, 其中博问频道关于"WPF", "C#", "WP8& ...