#15. 钻石

钻石

diamond.in/.out/.cpp

【问题描述】

你有 n 个 “量子态” 的盒子,每个盒子里可能是一些钱也可能是一个钻 石。

现在你知道如果打开第 i 个盒子,有100  Pi100 的概率能获得i  Vi 的钱,有1-100  Pi100 的概率能获得一个钻石。

现在你想知道,自己恰好获得 k(0 ≤ k ≤ n) 个钻石,并且获得钱数大 于等于 m 的概率是多少。

请你对 0 ≤ k ≤ n 输出 n+1 个答案。

答案四舍五入保留 3 位小数。

【输入格式】

第一行两个整数 n,m,见题意。

接下来 n 行,每行两个整数 i  Vi , i  Pi 。

【输出格式】

输出共 n+1 行,表示 0 ≤ k ≤ n 的答案。

【样例输入】

2 3

2 50

3 50

【样例输出】

0.250

0.250

0.000

【数据规模和约定】

对于 30% 的数据, n ≤ 10。

对于 60% 的数据, n ≤ 15

对于 100% 的数据,n ≤ 30, 1 ≤ i  Pi ≤ 99, 1 ≤ i  Vi ≤ 10^7 , 1 ≤ m ≤ 10^7 。

【限制】

时间:1s

内存:256M

 
同样是meet in the middle
为什么差距这么大呢?????
一个20ms+
一个1700ms+
=.=....
 
 1 #include<cstdio>
 2 #include<cstring>
 3 #include<algorithm>
 4 #include<vector>
 5 using namespace std;
 6 int tt;
 7 int n,m;
 8 int v[35];
 9 double p[35];
10 double ans[35];
11 vector<pair<int,double> > sta[35];
12 int main(){
13 
14     scanf("%d%d",&n,&m);
15     for(int i=1,x;i<=n;i++){
16         scanf("%d%d",&v[i],&x);
17         p[i]=x/100.;
18     }
19     for(int i=0;i<=n;i++){
20         sta[i].clear();
21     }
22     int an=(n/2.5)+1;
23     int bn=n-an;
24     for(int st=0;st<1<<bn;st++){
25         double nowp=1;
26         int cnt=0,money=0;
27         for(int i=0;i<bn;i++){
28             if((st>>i)&1){
29                 money+=v[n-i];
30                 nowp*=p[n-i];
31             }else{
32                 cnt++;
33                 nowp*=(1-p[n-i]);
34             }
35         }
36         sta[cnt].push_back(make_pair(money,nowp));
37     }
38     for(int i=0;i<=n;i++){
39         sort(sta[i].begin(),sta[i].end());
40         for(int j=1;j<sta[i].size();j++){
41             sta[i][j].second+=sta[i][j-1].second;
42         }
43     }
44     for(int st=0;st<1<<an;st++){
45         double nowp=1;
46         int cnt=0,money=0;
47         for(int i=0;i<an;i++){
48             if((st>>i)&1){
49                 money+=v[i+1];
50                 nowp*=p[i+1];
51             }else{
52                 cnt++;
53                 nowp*=(1-p[i+1]);
54             }
55         }
56         for(int i=0;i<=bn;i++){
57             // now d =cnt+i
58             int L = m-money;
59             vector<pair<int,double> >::iterator it = lower_bound(sta[i].begin(),sta[i].end(),make_pair(L,-1.));
60             double tmp = sta[i].back().second;
61             if(it!= sta[i].begin()){
62                 it--;
63                 tmp-=it->second;
64             }
65             ans[cnt+i] += tmp*nowp;
66         }
67     }
68     for(int i=0;i<=n;i++){
69         printf("%.3f\n",ans[i]);
70     }
71      fclose(stdout);
72     return 0;
73 }
20ms
 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 #include<cmath>
 5 #include<map>
 6 #include<algorithm>
 7 #define lli long long int 
 8 using namespace std;
 9 const int MAXN=88000;
10 inline void read(int &n)
11 {
12     char c=getchar();n=0;bool flag=0;
13     while(c<'0'||c>'9')    c=='-'?flag=1,c=getchar():c=getchar();
14     while(c>='0'&&c<='9')    n=n*10+c-48,c=getchar();flag==1?n=-n:n=n;
15 }
16 struct node
17 {
18     int p,v;
19 }box[MAXN];
20 double ans[MAXN];
21 int n,m;
22 int base;// 中间节点 
23 struct ANS1
24 {
25     int zuanshi;
26     int qian;
27     double gailv;
28 }ans1[MAXN];
29 int ans1tot=0;
30 struct ANS2
31 {
32     int zuanshi;
33     int qian;
34     double gailv;
35 }ans2[MAXN];
36 int ans2tot=0;
37 void dfs(int num,double gl,int money,int zs)
38 {
39     if(num==base)
40     {
41         ans1[++ans1tot].qian=money;
42         ans1[ans1tot].zuanshi=zs;
43         ans1[ans1tot].gailv=gl;
44         return ;
45     }
46     dfs(num+1,(double)gl*((double)box[num+1].p/100),money+box[num+1].v,zs);
47     dfs(num+1,(double)gl*(double)( 1 - ( (double) box[num+1].p/100) ),money,zs+1);
48 }
49 void dfs2(int num,double gl,int money,int zs)
50 {
51     if(num==n)
52     {
53         ans2[++ans2tot].qian=money;
54         ans2[ans2tot].zuanshi=zs;
55         ans2[ans2tot].gailv=gl;
56         return ;
57     }
58     dfs2(num+1,(double)gl*((double)box[num+1].p/100),money+box[num+1].v,zs);
59     dfs2(num+1,(double)gl*(double)( 1 - ( (double) box[num+1].p/100) ),money,zs+1);
60 }
61 int comp1(const ANS1 &a,const ANS1 &b)
62 {
63     return a.qian<b.qian;
64 }
65 int comp2(const ANS2 &a,const ANS2 &b)
66 {
67     return a.qian>b.qian;
68 }
69 int main()
70 {
71 
72     read(n);read(m);
73     base=n/2;
74     for(int i=1;i<=n;i++)
75     {
76         read(box[i].v);
77         read(box[i].p);
78     }
79     dfs(1,(double)box[1].p/100,box[1].v,0);//
80     dfs(1,(double)1-(double)box[1].p/100,0,1);// 钻石 
81     
82     dfs2(base+1,(double)box[base+1].p/100,box[base+1].v,0);//
83     dfs2(base+1,(double)1-(double)box[base+1].p/100,0,1);// 钻石 
84     
85     sort(ans1+1,ans1+ans1tot,comp1);
86     sort(ans2+1,ans2+ans2tot,comp2);
87     for(int i=1;i<=ans1tot;i++)
88     {
89         for(int j=1;j<=ans2tot;j++)
90         {
91             if(ans1[i].qian+ans2[j].qian>=m)
92                 ans[ans1[i].zuanshi+ans2[j].zuanshi]+=ans1[i].gailv*ans2[j].gailv;
93             else break;
94         }
95     }
96     for(int i=0;i<=n;i++)
97         printf("%.3lf\n",ans[i]);
98     return 0;
99 }
自己写的1700ms+

 

 
posted @ 2017-10-19 17:20  自为风月马前卒  阅读(215)  评论(0编辑  收藏  举报

Contact with me