[POJ3977] Subet(二分枚举)
解题报告
前置知识:折半查找法(二分法)
顾名思义,折半就是把一组数据(有序)分成两半,判断我们要找的key值在哪一半当中,不断重复该操作直至找到目标key值,这玩意说白了就是二分的另一个名字。
解决
一说二分大家都知道,但问题在于我们怎么往二分那里去想。首先打眼一看这道题不像一般的思维题那样有明显的规律,事实上也的确没有,那么我们初步的思路就只能是暴力枚举。
但单纯的暴力肯定是GG的,N<=35就意味着我们复杂度最多会达到2^35 这个级别,显然不现实。那么怎么减少枚举的次数呢,我们肯定会想到二分?那么为什么二分是可行的呢,我们如果把35个元素看作一个集合,然后把这个集合分成尽量均等的两份(为使复杂度尽量小,下面会说到),所以问题转化为从一个集合中找到若干集合(元素)使几个元素的绝对值最小,转变成求两个集合中的若干集合(元素)所组成的集合的绝对值最小。这里我们设这两个集合为A和B,简单表示一下就是:
- | 原集合中选若干个元素组成的子集 | = | A中选若干个元素 | + | B中选若干个元素 |
如果A中有numA个元素,那么A中的预选方案就有2^numA 种,这里可以用状态压缩的思想来解释。对于每一个元素只有选与不选两种状态,我们用一个二进制串来表示,1代表选0代表不选,那么总状态很显然有2^numA 种,B也是是一样的。
这样我们就可以对区间进行枚举了,使用整数表示集合,将所有可能的和与相对应的元素个数存入map,同时每遍历一个组合,就比较其是否比当前结果更优,如果是则更新结果。
然后对第二个区间进行枚举,每枚举出一种组合的和sum,比较更新结果。然后使用map内置的lower_bound函数,在第一个区间里找到比−sum大的最小元素,判断该元素与sum相加是否能构成更好的解。此时不能忘了,我们还要看比−sum小的最大元素,同样判断这种情况是否能构成更好的解。
我们是否需要对前半个区间取空集或者后半个区间取空集进行特判呢?好像没有关注过的样子?答案是不需要管它。因为我们无论是在前后区间进行枚举的时候,一旦找到一个sum值,就会判断它的绝对值是否比当前最优解更小,如果此时更新了结果值,也就是说我们只在单个区间里取了这些个元素,另一个区间根本没有枚举。已经考虑了空的情况。
实现
自己的代码有点小问题。。。先放上一个正解的,自己的等考完了再改改。。。(记得每次要清空map)
#include<iostream>
#include<iomanip>
#include<vector>
#include<algorithm>
#include<map>
#include<queue>
#include<string.h>
#include<math.h>
using namespace std;
#define ll long long
#define inf 1e9
#define MAX 100000
#define pair pair<ll,ll>
#define abs(x) ((x)>=0?(x):-(x))//手写的取绝对值函数,虽然不知道为什么但必须手写
ll n, a[40];
pair ans;
map<ll, ll > p;//value->len
map<ll, ll>::iterator it;//迭代器,都多久没用过了...
void solve() {
//折半枚举
for (ll i = 1; i < (1 << (n / 2)); i++) {//枚举前(n/2)位
ll t = i, sum = 0, len = 0;
for (ll j = n / 2 - 1; j >= 0; j--) {
if (t&(1 << j)) { sum += a[j]; len++; }
}
ll tmp = abs(sum);
if (tmp < ans.first || (tmp == ans.first&&len < ans.second))
ans = make_pair(tmp, len);
p[sum] > 0 ? p[sum] = min(p[sum], len) : p[sum] = len;
}
for (ll i = 1; i < (1 << (n - n / 2)); i++) {//枚举后(n/2)位 考虑空集
ll t = i, sum = 0, len = 0;
for (ll j = n - 1; j >= n / 2; j--) {
ll v = j - n / 2;
if (t&(1 << v)) { sum += a[j]; len++; }
}
ll tmp = abs(sum);
if (tmp < ans.first || (tmp == ans.first&&len < ans.second))
ans = make_pair(tmp, len);
it = p.lower_bound(-sum);//才知道map还自带lower_bound
if (it != p.end()) {//这是一个最接近他的大于它的相反数的元素
ll val = abs((*it).first + sum), l = (*it).second + len;
if (ans.first > val || (ans.first == val && l < ans.second)) {
ans.first = val; ans.second = l;
}
}
if (it != p.begin()) {//这是一个最接近他的小于它的相反数的元素
it--;
ll val = abs((*it).first + sum), l = (*it).second + len;
if (ans.first > val || (ans.first == val && l < ans.second)) {
ans.first = val; ans.second = l;
}
}
}
cout << ans.first << " " << ans.second << endl;
return;
}
int main(){
ios::sync_with_stdio(0);
while (cin >> n) {
if (n == 0) return 0;
p.clear();//每次清map
for (ll i = 0; i < n; i++) { cin >> a[i];}
ans = make_pair(abs(a[0]), 1);
solve();
}
}
[POJ3977] Subet(二分枚举)的更多相关文章
- FZU-2216 The Longest Straight (二分枚举)
题目大意:给n个0~m之间的数,如果是0,那么0可以变为任意的一个1~m之间的一个数.从中选出若干个数,使构成一个连续的序列.问能构成的最长序列的长度为多少? 题目分析:枚举连续序列的起点,二分枚举二 ...
- uva 12587 二分枚举
思路:维护一个森林,二分枚举最小的最大值. #include<set> #include<map> #include<cmath> #include<queu ...
- SDIBT 3237 Boring Counting( 划分树+二分枚举 )
http://acm.sdibt.edu.cn/JudgeOnline/problem.php?id=3237 Problem H:Boring Counting Time Limit: 3 Sec ...
- POJ 3273 Monthly Expense 二分枚举
题目:http://poj.org/problem?id=3273 二分枚举,据说是经典题,看了题解才做的,暂时还没有完全理解.. #include <stdio.h> #include ...
- POJ 2112 Optimal Milking(Floyd+多重匹配+二分枚举)
题意:有K台挤奶机,C头奶牛,每个挤奶机每天只能为M头奶牛服务,下面给的K+C的矩阵,是形容相互之间的距离,求出来走最远的那头奶牛要走多远 输入数据: 第一行三个数 K, C, M 接下来是 ...
- hdu 5248 序列变换(二分枚举)
Problem Description 给定序列A={A1,A2,...,An}, 要求改变序列A中的某些元素,形成一个严格单调的序列B(严格单调的定义为:Bi<Bi+,≤i<N). 我们 ...
- HDU 1669 Jamie's Contact Groups(多重匹配+二分枚举)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1669 题目大意: 给你各个人可以属于的组,把这些人分组,使这些组中人数最多的组人数最少,并输出这个人数 ...
- Codeforces 807C - Success Rate(二分枚举)
题目链接:http://codeforces.com/problemset/problem/807/C 题目大意:给你T组数据,每组有x,y,p,q四个数,x/y是你当前提交正确率,让你求出最少需要再 ...
- Codeforces 801C Voltage Keepsake(二分枚举+浮点(模板))
题目链接:http://codeforces.com/contest/801/problem/C 题目大意:给你一些电器以及他们的功率,还有一个功率一定的充电器可以给这些电器中的任意一个充电,并且不计 ...
- BFS+状态压缩DP+二分枚举+TSP
http://acm.hdu.edu.cn/showproblem.php?pid=3681 Prison Break Time Limit: 5000/2000 MS (Java/Others) ...
随机推荐
- TZOJ 车辆拥挤相互往里走
102路公交车是crq经常坐的,闲来无聊,他想知道最高峰时车上有多少人,他发现这辆车只留一个门上下人,于是他想到了一个办法,上车时先数一下车上人员数目(crq所上的站点总是人不太多),之后就坐在车门口 ...
- DMR对讲机利用XLX网络联网通信
By 申建军 BD8SN 2018-9-29 本文适用于运行G4KLX DMRGateway的MMDVM热点和中继,目前国内绝大部分的热点用户都是使用pi-star镜像的MMDVM热点,均可按此设置. ...
- 一文了解Docker容器技术的操作
一文了解Docker容器技术的操作 前言一.Docker是什么二.Docker的安装及测试Docker的安装Docker的Hello world测试三.Docker的常见操作镜像的基本操作容器的基本操 ...
- vue axios封装
前言: 对第三方库进行二次封装和抽离到统一模块,项目面对自己的模块进行开发.如果有一天更换库,只需要修改自己模块中的代码,无需对整个项目进行重构. 将axios网络请求库封装到network文件下的r ...
- 用于实现tab页签切换页面的angular路由复用策略
使用场景 打开菜单页面的时候,出现对应页面的页签.切换页签,原来的页面信息状态保留,关闭页签则保留的信息删除.使用路由复用策略,保存路由快照.实现效果如图所示 实现过程 概述: 1.在app.modu ...
- 【原创】Linux中断子系统(三)-softirq和tasklet
背景 Read the fucking source code! --By 鲁迅 A picture is worth a thousand words. --By 高尔基 说明: Kernel版本: ...
- Android学习笔记通过Toast显示消息提示框
显示消息提示框的步骤 这个很简单我就直接上代码了: Button show = (Button)findViewById(R.id.show); show.setOnClickListener(new ...
- vc6.0代码转vs2017相关问题
vc6.0代码转vs2017相关问题 命令行 error D8016: “/ZI”和“/Gy-”命令行选项不兼容fatal error C1083: 无法打开包括文件: “WinSock2.h”: N ...
- Hystrix Stream的监控页面不显示内容
打开Hystrix Stream页面,进入后,发现只有一行Unable to connect to Command Metric Stream. 因为springboot的默认路径不是 "/ ...
- IWAB0398E Error in generating WSDL from Java: java.lang.ClassNotFoundException
今天想用Eclipse创建WebService,报错信息 IWAB0398E Error in generating WSDL from Java: java.lang.ClassNotFoundEx ...