【折半枚举+二分】POJ 3977 Subset
题目内容
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的更多相关文章
- CSU OJ PID=1514: Packs 超大背包问题,折半枚举+二分查找。
1514: Packs Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 61 Solved: 4[Submit][Status][Web Board] ...
- poj 3977 Subset(折半枚举+二进制枚举+二分)
Subset Time Limit: 30000MS Memory Limit: 65536K Total Submissions: 5721 Accepted: 1083 Descripti ...
- POJ 3977 Subset(折半枚举+二分)
SubsetTime Limit: 30000MS Memory Limit: 65536KTotal Submissions: 6754 Accepted: 1277 D ...
- POJ 3977 - subset - 折半枚举
2017-08-01 21:45:19 writer:pprp 题目: • POJ 3977• 给定n个数,求一个子集(非空)• 使得子集内元素和的绝对值最小• n ≤ 35 AC代码如下:(难点:枚 ...
- POJ 3977 Subset
Subset Time Limit: 30000MS Memory Limit: 65536K Total Submissions: 3161 Accepted: 564 Descriptio ...
- POJ 3977:Subset(折半枚举+二分)
[题目链接] http://poj.org/problem?id=3977 [题目大意] 在n个数(n<36)中选取一些数,使得其和的绝对值最小. [题解] 因为枚举所有数选或者不选,复杂度太高 ...
- Subset POJ - 3977(折半枚举+二分查找)
题目描述 Given a list of N integers with absolute values no larger than 10 15, find a non empty subset o ...
- POJ - 3977 Subset(二分+折半枚举)
题意:有一个N(N <= 35)个数的集合,每个数的绝对值小于等于1015,找一个非空子集,使该子集中所有元素的和的绝对值最小,若有多个,则输出个数最小的那个. 分析: 1.将集合中的元素分成两 ...
- POJ 2549 Sumsets(折半枚举+二分)
Sumsets Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 11946 Accepted: 3299 Descript ...
随机推荐
- Apache报错:无法使用可靠的服务器域名
Apache 安装和启动时报错:无法使用可靠的服务器域名,打开Apache配置文件httpd.conf,去除 ServerName 前面的注释即可 1. 报错信息:无法使用可靠的服务器域名 AH005 ...
- JAVA热部署,通过agent进行代码增量热替换!!!
在前说明:好久没有更新博客了,这一年在公司做了好多事情,包括代码分析和热部署替换等黑科技,一直没有时间来进行落地写出一些一文章来,甚是可惜,趁着中午睡觉的时间补一篇介绍性的文章吧. 首先热部署的场景是 ...
- CTF-BugKu-加密
2020.09.12 恕我直言,上午做WeChall那个做自闭了,下午复习一下之前做过的. 做题 第一题 滴答~滴 https://ctf.bugku.com/challenges#滴答~滴 摩斯密码 ...
- docker部署LAMP架构并部署上线wordpress博客系统
第一步:直接在镜像仓库拉取LAMP镜像 [root@ken-node3 ken]# docker pull tutum/lamp 第二步:查看已经获取到的镜像 [root@ken-node3 ken] ...
- 【转】Postgres SQL sort 操作性能调优
这篇文章将以实战的方式结合笔者在项目中真实遇到的情况来讲解.说到SQL,大家可能会遇到一些写法稍微复杂的写法.比如SQL中遇到的有聚合函数sum等,也有遇到使用group by / order by的 ...
- Android如何使用注解进行代码检查
原文首发于微信公众号:躬行之(jzman-blog),欢迎关注交流! Android Studio 内置了代码检查工具 Lint,可在菜单栏选择 Analyze > Inspect Code 执 ...
- 802.1X 账号密码+设备信息双重认证
名词解释 802.1X: IEEE802 LAN/WAN 委员会为解决无线局域网网络安全问题,提出了 802.1X 协议.后来,802.1X协议作为局域网端口的一个普通接入控制机制在以太网中被广泛应用 ...
- 记一次select2赋值动态数组的坑
var roles = $td.eq(3).text().split(","); var arr = []; //循环去除每个值前后的空格,否则下拉框赋值回显出错for(var i ...
- MySQL存储引擎入门介绍
什么是MySQL? MySQL 是一种关系型数据库,在Java企业级开发中非常常用,因为 MySQL 是开源免费的,并且方便扩展.阿里巴巴数据库系统也大量用到了 MySQL,因此它的稳定性是有保障的. ...
- C++练习案例1.计算机类(利用多态实现)
c++简单计算机类 简介 大家好,这里是天天like的博客,这是我发的第一篇随笔,用来记录我的学习日程,大家可以相互学习,多多交流,感谢 今天我要记录的随笔是在学习c++多态的知识点练习改进的一个案例 ...