poj3977 折半枚举

传送门:https://vjudge.net/problem/POJ-3977

题意:给你n数(n<=35),从中选出一个非空子集,使得这个子集的所有元素的值的和的绝对值最小,如果有多组数据满足的话,选择子集元素最少的那个。

这题是从挑战程序设计竞赛来的。就是折半枚举。也就是我先分别枚举前面一半的选不选状态(最多2的17次方,O(能过)),和后面一半的,得到两个子集和的数列。子集要么是只从前面一半选,要么只从后面一半选,要么从前面和后面。前面两种情况可以在枚举的时候搞了。至于最后那种情况,我们可以这样解决:就是先对两个枚举出来的子集和数列排个序,然后枚举第一个数列,对于枚举的第一个数列的子集和sum,我们要让它加上从第二个数列的子集和的绝对值最小,也就是在第二个数列中找-sum。这个由于排过序了,所以可以二分。nlogn,可过。

要注意的地方是:首先,不能是空集,所以每次更新答案前要判断它是否空。

其次:我在找第二个数列找-sum的时候,用lower_bound找到p,但是可能找到的数比-sum大的,所以还要比较一下p-1那个数。具体看代码吧。

 

还有,这题再次验证了一个定律。就是每当我卡题的时候,只要我说一句“不过hjj××”,这题就一定AC。这题就是最好的证明了。。。。。

 1 // Cease to struggle and you cease to live
 2 #include <iostream>
 3 #include <cmath>
 4 #include <cstdio>
 5 #include <cstring>
 6 #include <algorithm>
 7 #include <queue>
 8 #include <vector>
 9 #include <set>
10 #include <map>
11 #include <stack>
12 using namespace std;
13 typedef long long ll;
14 ll a[40];
15 #define pii pair<ll,int> 
16 #define mpr make_pair
17 pii num1[(int)3e5],num2[(int)3e5];
18 ll _abs(ll a){return a>0?a:-a;}
19 int main() {
20     int n;
21     while(~scanf("%d",&n)){
22         if(!n) break;
23         pii ans=mpr(100000000000000000,0x3f3f3f3f);
24         for(int i=1;i<=n;++i) scanf("%lld",&a[i]);
25         int a1=n/2,a2=n-a1;
26         for(int i=0;i<(1<<a1);++i){
27             num1[i].first=num1[i].second=0;
28             for(int j=1;j<=a1;++j){
29                 if((i>>(a1-j))&1){
30                     num1[i].first+=a[j],++num1[i].second; 
31                 }
32             }
33             pii tem=mpr(_abs(num1[i].first),num1[i].second);
34             if(tem.second && tem<ans) ans=tem;
35         }
36         for(int i=0;i<(1<<a2);++i){
37             num2[i].first=num2[i].second=0;
38             for(int j=1;j<=a2;++j){
39                 if((i>>(a2-j))&1) num2[i].first+=a[j+a1],++num2[i].second;
40             }
41             pii tem=mpr(_abs(num2[i].first),num2[i].second);
42             if(tem.second && tem<ans) ans=tem;
43         }
44         //cerr<<ans.first<<ans.second<<endl;
45         sort(num1,num1+(1<<a1));
46         sort(num2,num2+(1<<a2));
47         for(int i=0;i<(1<<a1);++i){
48             ll c=num1[i].first;
49             int p=lower_bound(num2,num2+(1<<a2),mpr(-c,-1))-num2;
50             if(p!=0){
51                 pii tem=mpr(_abs(num1[i].first+num2[p-1].first),num1[i].second+num2[p-1].second);
52                 if(tem.second && tem<ans) ans=tem;
53             }
54             //cerr<<ans.first<<endl;
55             if(p!=(1<<a2)){
56                 pii tem=mpr(_abs(num1[i].first+num2[p].first),num1[i].second+num2[p].second);
57                 //cerr<<num1[i].first<<' '<<num2[p].first<<' '<<p<<endl;
58                 if(tem.second && tem<ans) ans=tem;
59             }
60             //cerr<<ans.first<<endl;
61         }
62         printf("%lld %d\n",ans.first,ans.second);
63     }
64     return 0;
65 }
View Code

最近训练的有些松散,接下来一定勤加练习。

 

睿。

posted @ 2019-05-11 22:45  小布鞋  阅读(230)  评论(0编辑  收藏  举报