牛客网 网易2019实习生招聘编程题第八题 子集生成+二分

https://www.nowcoder.com/test/9763997/summary

牛牛准备参加学校组织的春游, 出发前牛牛准备往背包里装入一些零食, 牛牛的背包容量为w。
牛牛家里一共有n袋零食, 第i袋零食体积为v[i]。
牛牛想知道在总体积不超过背包容量的情况下,他一共有多少种零食放法(总体积为0也算一种放法)。

 

输入描述:
输入包括两行
第一行为两个正整数n和w(1 <= n <= 30, 1 <= w <= 2 * 10^9),表示零食的数量和背包的容量。
第二行n个正整数v[i](0 <= v[i] <= 10^9),表示每袋零食的体积。



输出描述:
输出一个正整数, 表示牛牛一共有多少种零食放法。

 

输入例子1:
3 10
1 2 4

 

输出例子1:
8

 

例子说明1:
三种零食总体积小于10,于是每种零食有放入和不放入两种情况,一共有2*2*2 = 8种情况。

解析 我们一半一半枚举 枚举前一半 把每种情况的体积和存到数组ans里 再枚举后一半 当枚举到一种情况的体积和为sum时 二分查找数组ans中值小于等于w-sum的数量 加到答案里

AC代码
 1 #include <bits/stdc++.h>
 2 using namespace std;
 3 typedef long long ll;
 4 const ll mod=1000000007,maxn=5e4+50;
 5 vector<ll> a,ans;
 6 ll n,m;
 7 int main()
 8 {
 9     while(cin>>n>>m)
10     {
11         a.clear(),ans.clear();
12         for(int i=0; i<n; i++)
13         {
14             ll x;
15             cin>>x,a.push_back(x);
16         }
17         ll n1=n>>1,n2=n-n1;
18         for(int i=0; i<(1<<n1); i++)
19         {
20             ll sum=0;
21             for(int j=0; j<n1; j++)
22                 if(i&(1<<j)) sum+=a[j];
23             ans.push_back(sum);
24         }
25         sort(ans.begin(),ans.end());
26         ll anss=0;
27         for(int i=0; i<(1<<n2); i++)
28         {
29             ll sum=0;
30             for(int j=0; j<n2; j++)
31             {
32                 if(i&(1<<j))
33                     sum+=a[n1+j];
34             }
35             anss+=lower_bound(ans.begin(),ans.end(),m-sum+1)-ans.begin();
36         }
37         cout<<anss<<endl;
38     }
39 }

 

posted @ 2018-05-21 21:19  灬从此以后灬  阅读(311)  评论(0编辑  收藏  举报