题目内容

Vjudge链接

给你\(n\)个数,求出这\(n\)个数的一个非空子集,使子集中的数加和的绝对值最小,在此基础上子集中元素的个数应最小。

输入格式

输入含多组数据,每组数据有两行,第一行是元素组合\(n\)(若\(n\)为0表示输入结束),第二行有\(n\)个数,表示要给出的\(n\)个数。

数据范围

\(n\le 35\)

输出格式

每组数据输出一行两个数中间用空格隔开,表示最小的绝对值和该子集的元素个数。

样例输入

1

10

3

20 100 -100

0

样例输出

10 1

0 2

思路

在学必修一的第一节课我们就知道有\(n\)个数的集合的非空子集有\(2^n-1\)个,因此此题直接枚举的话大小回到\(2^{35}-1\),显然出题人没有这么好心。

这里用到一种思想叫做对半枚举。可以想到\(2^{17}-1=131071\),比较优秀的,那么就开始对半枚举就好了。将\(n\)的数分为前\(\frac{n}{2}\)和后\(\frac{n}{2}\)进行枚举。同时可以结合二分查找。

如果是一个空集加上一个非空,那么得到的也是非空集合,当然如果两个空集加在一起肯定也还是空集。

注意下边界。

代码

#include <cstdio>
#include <map>
#include <algorithm>
using namespace std;
typedef long long ll; ll Abs(ll x){//据说POJ不支持abs函数?
return x > 0 ? x : -x;
} int n;
ll a[40];
int main(){
while (scanf("%d",&n)&&n!=0){
for (int i=1;i<=n;++i)
scanf("%d",a+i);
map<ll,int> g;//一开始想用结构体结果发现结构体不能用lower_bound
ll ans=Abs(a[1]);
int len=n; for (int i=1;i<(1<<(n/2));++i){
ll sum=0;
int j=i,cnt=0,pos=1;
while(j&&pos<=n/2){
if (j&1){
sum+=a[pos];
cnt++;
}
j>>= 1;
pos++;
}
if (Abs(sum)<ans){
ans=Abs(sum);
len=cnt;
}
else if(Abs(sum)==ans){
len=min(len, cnt);
} if(g[sum])
g[sum]=min(g[sum], cnt);
else
g[sum]=cnt;
} for (int i=1;i<(1<<(n-n/2));++i){
ll sum=0;
int cnt=0,pos=1,j=i;
while (j&&pos+n/2<= n){
if (j&1){
sum+= a[pos+n/2];
cnt++;
}
j>>= 1;
pos++;
} if(Abs(sum) < ans){
ans=Abs(sum);
len=cnt;
}
else if(Abs(sum)==ans){
len=min(len,cnt);
} map<ll,int>::iterator it=g.lower_bound(-sum);//迭代器都要忘光了orz if(it!=g.end()){
if(Abs(sum+it->first)<ans){
ans=Abs(sum+it->first);
len=cnt+it->second;
}
else if(Abs(sum+it->first)==ans){
len=min(len,cnt+it->second);
}
} if (it!=g.begin()){
it--;
if(Abs(sum+it->first)<ans){
ans=Abs(sum+it->first);
len=cnt+it->second;
}
else if(Abs(sum+it->first)==ans){
len=min(len,cnt+it->second);
}
}
} scanf("%d %d\n",Abs(ans),len);
}
return 0;
}

代码启发

https://www.jianshu.com/p/27eefa7b990e

膜一下dalao的玛丽

【折半枚举+二分】POJ 3977 Subset的更多相关文章

  1. CSU OJ PID=1514: Packs 超大背包问题,折半枚举+二分查找。

    1514: Packs Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 61  Solved: 4[Submit][Status][Web Board] ...

  2. poj 3977 Subset(折半枚举+二进制枚举+二分)

    Subset Time Limit: 30000MS   Memory Limit: 65536K Total Submissions: 5721   Accepted: 1083 Descripti ...

  3. POJ 3977 Subset(折半枚举+二分)

    SubsetTime Limit: 30000MS        Memory Limit: 65536KTotal Submissions: 6754        Accepted: 1277 D ...

  4. POJ 3977 - subset - 折半枚举

    2017-08-01 21:45:19 writer:pprp 题目: • POJ 3977• 给定n个数,求一个子集(非空)• 使得子集内元素和的绝对值最小• n ≤ 35 AC代码如下:(难点:枚 ...

  5. POJ 3977 Subset

    Subset Time Limit: 30000MS   Memory Limit: 65536K Total Submissions: 3161   Accepted: 564 Descriptio ...

  6. POJ 3977:Subset(折半枚举+二分)

    [题目链接] http://poj.org/problem?id=3977 [题目大意] 在n个数(n<36)中选取一些数,使得其和的绝对值最小. [题解] 因为枚举所有数选或者不选,复杂度太高 ...

  7. Subset POJ - 3977(折半枚举+二分查找)

    题目描述 Given a list of N integers with absolute values no larger than 10 15, find a non empty subset o ...

  8. POJ - 3977 Subset(二分+折半枚举)

    题意:有一个N(N <= 35)个数的集合,每个数的绝对值小于等于1015,找一个非空子集,使该子集中所有元素的和的绝对值最小,若有多个,则输出个数最小的那个. 分析: 1.将集合中的元素分成两 ...

  9. POJ 2549 Sumsets(折半枚举+二分)

    Sumsets Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 11946   Accepted: 3299 Descript ...

随机推荐

  1. MvvmLight框架使用入门(5)

    上一次写MvvmLight框架使用入门(4)的时候还在用Visual Studio 2015,我儿子也不会过来盖上我的XPS……重启这个系列一方面是因为最近又开始写UWP的东西了,另一个是因为Mvvm ...

  2. tcp建立连接为什么需要三次握手和四次挥手

    前言 众所周知tcp传输层协议在建立连接的时候需要三次才能建立起一个真正的可靠连接,可是为什么是三次呢,不可以是两次,四次等等呢,可以自己思考一番,带着疑问可以看下文. 三次握手 在<计算机网络 ...

  3. 源码解读 TDengine 中线程池的实现

    这篇文章中提到了 tsched 的源码可以一读,所以去阅读了一下,总共220来行. 1. 阅读前工作 通过上文了解到这段程序实现的是一个任务队列,同时带有线程池.这段程序是计算机操作系统里经典的con ...

  4. 2020JavaWeb之宝塔安装tomcat+nginx关于jsp处理问题

    关于nginx反向代理,是将jsp文件转交给tomcat处理,nginx主要处理静态资源,nginx处理静态资源的效率相对于tomcat高的多 在配置文件如下部分: location ~ \.jsp$ ...

  5. elasticsearch 索引清理脚本及常用命令

    elastic索引日志清理不及时,很容易产生磁盘紧张,官网给出curl -k -XDELETE可以清理不需要的索引日志. 清理脚本 #!/bin/bash #Author: 648403020@qq. ...

  6. JAVA8--流处理和optional

    转载自https://www.cnblogs.com/invoker-/p/6896865.html 流与集合    众所周知,日常开发与操作中涉及到集合的操作相当频繁,而java中对于集合的操作又是 ...

  7. PG-跨库操作-postgres_fdw

    接上一篇<PG-跨库操作-dblink>:讲下postgres_fdw的使用:postgres_fdw工作原理详细介绍可以去看下<PostgreSQL指南>第4章: 对FDW特 ...

  8. python类中的__init__和__new__方法

    Python中类: Python中在创建类的过程中最先调用的不是__init__方法而是__new__方法,__new__方法是一个静态方法,在创建一个类对象时其实是通过__new__方法首先创建出一 ...

  9. Apollo系列(二):Apollo在ASP.NET Core 3.1中使用

    关于Apollo怎么安装,我就不介绍,可以看这篇文章:https://www.cnblogs.com/vic-tory/p/13736192.html 一.Apollo使用: 1.创建项目 2.添加配 ...

  10. Generator函数在流程控制中的应用

    扯蛋 做了两年的Nodejs全栈开发,不知道为什么跑来做游戏呢(大概是厦门nodejs不好找工作吧).用的是网易的pomelo的游戏框架.现接手了一个棋牌游戏:二十一点,不懂的规则的可以自行百度. 二 ...