解题报告

前置知识:折半查找法(二分法)

顾名思义,折半就是把一组数据(有序)分成两半,判断我们要找的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(二分枚举)的更多相关文章

  1. FZU-2216 The Longest Straight (二分枚举)

    题目大意:给n个0~m之间的数,如果是0,那么0可以变为任意的一个1~m之间的一个数.从中选出若干个数,使构成一个连续的序列.问能构成的最长序列的长度为多少? 题目分析:枚举连续序列的起点,二分枚举二 ...

  2. uva 12587 二分枚举

    思路:维护一个森林,二分枚举最小的最大值. #include<set> #include<map> #include<cmath> #include<queu ...

  3. SDIBT 3237 Boring Counting( 划分树+二分枚举 )

    http://acm.sdibt.edu.cn/JudgeOnline/problem.php?id=3237 Problem H:Boring Counting Time Limit: 3 Sec  ...

  4. POJ 3273 Monthly Expense 二分枚举

    题目:http://poj.org/problem?id=3273 二分枚举,据说是经典题,看了题解才做的,暂时还没有完全理解.. #include <stdio.h> #include ...

  5. POJ 2112 Optimal Milking(Floyd+多重匹配+二分枚举)

    题意:有K台挤奶机,C头奶牛,每个挤奶机每天只能为M头奶牛服务,下面给的K+C的矩阵,是形容相互之间的距离,求出来走最远的那头奶牛要走多远   输入数据: 第一行三个数 K, C, M  接下来是   ...

  6. hdu 5248 序列变换(二分枚举)

    Problem Description 给定序列A={A1,A2,...,An}, 要求改变序列A中的某些元素,形成一个严格单调的序列B(严格单调的定义为:Bi<Bi+,≤i<N). 我们 ...

  7. HDU 1669 Jamie's Contact Groups(多重匹配+二分枚举)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1669 题目大意: 给你各个人可以属于的组,把这些人分组,使这些组中人数最多的组人数最少,并输出这个人数 ...

  8. Codeforces 807C - Success Rate(二分枚举)

    题目链接:http://codeforces.com/problemset/problem/807/C 题目大意:给你T组数据,每组有x,y,p,q四个数,x/y是你当前提交正确率,让你求出最少需要再 ...

  9. Codeforces 801C Voltage Keepsake(二分枚举+浮点(模板))

    题目链接:http://codeforces.com/contest/801/problem/C 题目大意:给你一些电器以及他们的功率,还有一个功率一定的充电器可以给这些电器中的任意一个充电,并且不计 ...

  10. BFS+状态压缩DP+二分枚举+TSP

    http://acm.hdu.edu.cn/showproblem.php?pid=3681 Prison Break Time Limit: 5000/2000 MS (Java/Others)   ...

随机推荐

  1. Linux 用户组管理命令

    groupadd 组名,可以添加用户组 groupmod -n 新组名 老组名,可以修改组名 groupdel 组名,可以删除组(组中不能有初始用户存在,附加用户无所谓) gpasswd -a 用户名 ...

  2. STM32F429时钟不正确导致串口无法正确收发

    老早之前自己做了块F4的板子,设计原理图时没有去找官方参考,看了手册后就开工了,做完板子回来测试串口发现PC端接收到的都是乱码,尝试了几种波特率也没能正确接收,串口的代码是官方参考例程的,不应该有问题 ...

  3. 记录RecyclerView的位置并进行恢复

    //监听RecyclerView滚动状态 mRecyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() { @Overri ...

  4. CSS学习:overflow:hidden解决溢出,坍塌,清除浮动

    overflow:hidden是overflow属性的一个神奇用法,它可以帮助我们隐藏溢出的元素,清除浮动和解除坍塌. CSS样式: .container{ background-color: bla ...

  5. @atcoder - AGC026F@ Manju Game

    目录 @description@ @solution@ @accepted code@ @details@ @description@ 给定一个含 N 个数的序列,Alice 与 Bob 在博弈.Al ...

  6. 将MySQL库的表转入到MSSQL中的某个库中(Employees下的Employees表 → pubs库下)_2

    将MySQL库的表转入到MSSQL中的某个库中(Employees下的Employees表 → pubs库下, 此pubs下的表名是employee,不冲突),方法大致以下几个(另有其他方法待补充), ...

  7. 全网最完整的Redis入门指导

    前言 本文提供全网最完整的Redis入门指导教程,下面我们从下载Redis安装包开始,一步一步的学习使用. 下载Redis 官网提供的Redis安装包是服务于Linux的,而我们需要在Window下使 ...

  8. 面试必问:分布式锁实现之zk(Zookeeper)

    点赞再看,养成习惯,微信搜索[三太子敖丙]关注这个互联网苟且偷生的工具人. 本文 GitHub https://github.com/JavaFamily 已收录,有一线大厂面试完整考点.资料以及我的 ...

  9. springboot整合Mybatis(有xml)

    pom.xml <!-- mybatis 支持 SpringBoot --> <dependency> <groupId>org.mybatis.spring.bo ...

  10. JVM 之 Linux定位CPU过高问题及优化

    项目部署以后出行卡顿现象,所以对问题进行了排查,记录一下排查过程 (从CSDN编辑器贴过来的,图有水印) 1.找进程 top 可以发现,是Java进程导致的CPU过高,致使系统卡顿 2.找线程 ps ...