# poj3977 Subset 折半枚举

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cmath>
using namespace std;
struct Z{
long long int x;
int y;
bool operator < (const Z& b)const

{
if(x!=b.x)
return x < b.x;
return y<b.y;

}
}a[300005],b[300005];
long long  int c[40];
long long int abs1(long long int x){
if(x<0)
return -x;
return x;
}
int main(){
int n;
while((cin>>n)&&n!=0){
for(int i=0;i<300005;i++)
a[i].x=a[i].y=b[i].x=b[i].y=0;
long long int sum=1e17;
int ans=40;
for(int i=0;i<n;i++)
cin>>c[i];
int n1=n/2;
for(int i=0;i<1<<n1;i++){
for(int j=0;j<n1;j++){
if(i>>j&1&&(i!=0||j!=0)){
a[i-1].x+=c[j];
a[i-1].y++;
}
}

}
int n2=n-n1;
for(int i=0;i<(1<<n2);i++){
for(int j=0;j<n2;j++){
if(i>>j&1&&(i!=0||j!=0)){
b[i-1].x+=c[j+n1];
b[i-1].y++;
}
}

}
for(int i=0;i<(1<<n1)-1;i++){
if(abs1(a[i].x)<sum){
sum=abs1(a[i].x);
ans=a[i].y;
}
else if(abs1(a[i].x)==sum&&a[i].y<ans){
ans=a[i].y;
sum=abs1(a[i].x);
}
}
for(int i=0;i<(1<<n1)-1;i++)
a[i].x=-a[i].x;
for(int i=0;i<(1<<n2)-1;i++){
if(abs1(b[i].x)<sum){
sum=abs1(b[i].x);
ans=b[i].y;
}
else if(abs1(b[i].x)==sum&&b[i].y<ans){
ans=b[i].y;
sum=abs1(b[i].x);
}
}

sort(a,a+(1<<n1)-1);
sort(b,b+(1<<n2)-1);
for(int i=0;i<(1<<n1)-1;i++){
int t=lower_bound(b,b+(1<<n2)-1,a[i])-b;
if(t>0){
if(abs1(b[t-1].x-a[i].x)<sum){
sum=abs1(b[t-1].x-a[i].x);
ans=b[t-1].y+a[i].y;
}
else if(abs1(b[t-1].x-a[i].x)==sum&&b[t-1].y+a[i].y<ans){
sum=abs1(b[t-1].x-a[i].x);
ans=b[t-1].y+a[i].y;
}
}
if(t<(1<<n2)-1){
if(abs1(b[t].x-a[i].x)<sum){
sum=abs1(b[t].x-a[i].x);
ans=b[t].y+a[i].y;
}
else if(abs1(b[t].x-a[i].x)==sum&&b[t].y+a[i].y<ans){
sum=abs1(b[t].x-a[i].x);
ans=b[t].y+a[i].y;
}
}

}
cout<<sum<<" "<<ans<<endl;
}

}

• 本文已收录于以下专栏：

## 折半枚举--poj3977

Language: Default Subset Time Limit: 30000MS   Memory Limit: 65536K Total Submis...

## poj3977

#include #include #include #include using namespace std; pair cor[1005]; int n,d,par[1005],rk[1005...

## POJ 3977Subset（枚举+二分）

Subset Time Limit: 30000MS   Memory Limit: 65536K Total Submissions: 1562   Accepted: 261 ...
• opm777
• 2014-05-20 17:43
• 1107

## 《挑战程序设计竞赛》3.2.4 常用技巧-折半枚举 POJ2785 3977 2549

POJ2785http://poj.org/problem?id=2785题意输入n，表示a b c d 四个集合都有n个元素。之后每行输入4个集合中的一个元素。求这四个集合每个集合中拿出一个数相加等...

举报原因： 您举报文章：深度学习：神经网络中的前向传播和反向传播算法推导 色情 政治 抄袭 广告 招聘 骂人 其他 (最多只允许输入30个字)