Lightoj 1127 - Funny Knapsack 【二分】
阿新 • • 發佈:2017-06-12
mod ava assert pri pen type case limits roc
題目鏈接:
problem=1127">http://www.lightoj.com/volume_showproblem.php?problem=1127
題意:有n個物體(n<30)和一個容量為W的容器。問將容器不裝滿的放置物品的方式有多少種。
思路 : 狀態壓縮+二分。將前n/2個物體看做一個總體,將剩下的看做一個總體。1<<(n/2)個狀態代表前一半的物品使用情況,然後求出每一種狀態的總的體積。排序。對於後面的那一半也是。答案僅僅需枚舉一半然後在還有一半中找和W差的下界就可以。
代碼:
#include <stdio.h>
#include <ctime>
#include <math.h>
#include <limits.h>
#include <complex>
#include <string>
#include <functional>
#include <iterator>
#include <algorithm>
#include <vector>
#include <stack>
#include <queue>
#include <set>
#include <map>
#include <list>
#include <bitset>
#include <sstream>
#include <iomanip>
#include <fstream>
#include <iostream>
#include <ctime>
#include <cmath>
#include <cstring>
#include <cstdio>
#include <time.h>
#include <ctype.h>
#include <string.h>
#include <assert.h>
using namespace std;
int n;
long long w;
long long a[1000000];
long long s1[1000000];
long long s2[1000000];
int main()
{
int t;
scanf("%d", &t);
int cases = 1;
while (t--)
{
memset(s1, 0, sizeof(s1));
memset(s2, 0, sizeof(s2));
memset(a, 0, sizeof(a));
scanf("%d %lld", &n, &w);
for (int i = 0; i < n; i++)
scanf("%lld", &a[i]);
int t1 = n >> 1;
int t2 = n - t1;
int r1 = 1 << t1;
int r2 = 1 << t2;
for (int i = 0; i < r1; i++)
for (int j = 0; j < t1; j++)
{
if (i &(1 << j))
s1[i] += a[j];
}
for (int i = 0; i < r2; i++)
for (int j = 0; j < t2; j++)
{
if (i &(1 << j))
s2[i] += a[t1 + j];
}
sort(s2, s2 + r2 );
long long ans = 0;
for (int i = 0; i < r1; i++)
{
if (w - s1[i] >= 0)
ans += upper_bound(s2, s2 + r2, w - s1[i]) - s2;
}
printf("Case %d: %lld\n", cases++, ans);
}
return 0;
}
Lightoj 1127 - Funny Knapsack 【二分】