HDU2955-Robberies-01背包

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2955

英文题就得有翻译。。。

题意:Roy想要抢劫银行,每家银行多有一定的金额和被抓到的概率,知道Roy被抓的最大概率P,求Roy在被抓的情况下,抢劫最多。
 
输入数据:T,表示多少组测试数据;
double P,int n;表示被抓的最大概率,和银行数量;
接下来n行,每行一个整数cost,一个小数v,分别表示该家银行可以偷cost数量的金币,被抓的概率为v;
 
分析:计算被抓率不容易,我们可以转化成计算安全率,即被抓概率转换成安全概率,Roy的安全概率大于1-P时都是安全的。抢劫的金额为0时,肯定是安全的,所以d[0]=1;其他金额初始为最危险的所以概率全为0;
#include<iostream>
#include<string>
#include<cstdio>
#include<cstring>
#include<map>
#include<queue>
#include<cmath>
#include<stack>
#include<set>
#include<vector>
#include<algorithm>
#define LL long long
#define inf 1<<29
#define s(a) scanf("%d",&a)
#define CL(a,b) memset(a,b,sizeof(a))
using namespace std;
const int N=10005;
int n,m,a,b,c;
double p;
double v[N];    //  保存单个逃跑的几率;
double dp[N];   //  保存盗取N的金币逃跑的几率;
int cost[N];    //  保存可获得的量;
int main()
{
    int t;
    s(t);
    while(t--){
        int sum=0;
        scanf("%lf%d",&p,&n);
        for(int i=0;i<n;i++){
            scanf("%d%lf",&cost[i],&v[i]);
            sum+=cost[i];       //  保存最多可以盗取的金币数量;
        }
        CL(dp,0);dp[0]=1.0;     //  盗取金币为零的时候逃跑的概率为1;
        for(int i=0;i<n;i++)    //  遍历每一个银行;
            for(int j=sum;j>=cost[i];j--)   //  判断取还是不取;
                dp[j]=max(dp[j],dp[j-cost[i]]*(1-v[i]));
        for(int i=sum;i>=0;i--)
            if(dp[i]-1+p>1e-8){
                printf("%d\n",i);
                break;
            }
    }
    return 0;
}


 

你可能感兴趣的:(01背包问题,hdu2955)