【JZOJ4841】【NOIP2016提高A组集训第4场11.1】平衡的子集
题目描述
夏令营有N个人,每个人的力气为M(i)。请大家从这N个人中选出若干人,如果这些人可以分成两组且两组力气之和完全相等,则称为一个合法的选法,问有多少种合法的选法?
数据范围
40%的数据满足:1<=M(i)<=1000;
对于100%的数据满足:2<=N<=20,1<=M(i)<=100000000
解法
40%
枚举每一位选或不选,设当前选的所有数的和为sum,然后使用背包求出当前每个可能的总和。
如果其中有sum/2,那么说明这种选法合法,使答案+1;
100%
比较显然的暴力是O(320)。
现在考虑使用折半搜索;
将原数集分为两个均匀的集合,
分别搜索出两个集合中所有可能的和,O(2∗310);
由于一个集合中的负系数相当于把这项移项到另一个集合中。
所以直接找出两个集合中可能的和相等的数量即可。
运用二进制记录每个和选的方案,来防止重复计数。
实现:
将两组和排序,设l为第一组的指针,r为第二组的指针;
通过此大彼进,使得a[l]==b[r];
相同的和之间两两匹配来贡献答案,判重。
代码
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<iostream>
using namespace std;
const char* fin="subset.in";
const char* fout="subset.out";
const int maxn=50,maxh=1048580;
const int er[21]={1,2,4,8,16,32,64,128,256,512,1024};
int n,nn,i,j,k;
int a[maxn],b[maxn];
bool bz[1025][1025];
int h[maxh];
int ans1;
int fi[maxh],ne[maxh],la[maxh],en[maxh];
int ans,tot,tot1;
int ans2;
void add_line(int a,int b,int c){
if (h[a]<0){
tot++;
fi[a]=en[a]=tot;
la[tot]=b;
h[a]=c;
}else{
ne[en[a]]=tot+1;
tot++;
en[a]=tot;
la[tot]=b;
}
}
int hash(int v){
int k=(v%maxh+maxh)%maxh;
while (h[k]!=ans1 && h[k]!=v) {
k=(k+1)%maxh;
}
return k;
}
void dfs(int v,int sum,int state){
int i,j,k;
if (v>nn){
k=hash(sum);
add_line(k,state,sum);
return ;
}
dfs(v+1,sum,state);
dfs(v+1,sum+a[v],state+er[v-1]);
dfs(v+1,sum-a[v],state+er[v-1]);
}
void getans(int v,int sum,int stat){
int i,j,k;
if (v>n){
k=hash(sum);
if (h[k]!=ans1){
if (fi[k]==ans1) return;
ans2++;
j=0;
for (i=fi[k];i;i=ne[i]){
j++;
if (!bz[stat][la[i]]){
ans++;
bz[stat][la[i]]=true;
}
}
//ans2=ans2+j;
}
return;
}
getans(v+1,sum,stat);
getans(v+1,sum+a[v],stat+er[v-1-nn]);
getans(v+1,sum-a[v],stat+er[v-nn-1]);
}
int main(){
freopen(fin,"r",stdin);
freopen(fout,"w",stdout);
scanf("%d",&n);
nn=n/2;
memset(h,128,sizeof(h));
memset(fi,128,sizeof(fi));
ans1=h[0];
for (i=1;i<=n;i++) scanf("%d",&a[i]);
dfs(1,0,0);
getans(nn+1,0,0);
ans--;
printf("%d",ans);
return 0;
}
启发
折半搜索
本题
本题:每个数有不选,隶属A,隶属B,三种选择。
而对于在一个集合中,每个数的选择意义就是不选,隶属这个集合①,隶属另一个集合②。
①②选择在两个集合之间互逆。
姑且称这些元素的抉择是奇性的。
那么本体就可以使用折半搜索。
拓展
现在有这么一题,
夏令营有N个人,每个人的力气为M(i)。问有多少种分法将这N个人分成两组且两组力气之和完全相等?(n<=40,m(i)<=10000000)
暂且不考虑其他玄学算法,考虑折半搜索:
直接的想法是O(2n)搜索,这显然超时;
但每个项的两个抉择互逆,也就是说每个项的抉择呈奇性。
将原数集分为两个均匀的集合。
分别使用O(2n/2)处理出两个集合的所有可能的和。
然后合并答案即可,使用二进制判重。
上述脑洞题算是与原问题的一个“外传”,也是应用折半搜索。
总结
大概折半搜索需要这个条件,将原数集分成两个均匀的集合后,项的抉择呈奇性。
至于奇性是什么上文已感性阐述,属博主自创词汇。
【JZOJ4841】【NOIP2016提高A组集训第4场11.1】平衡的子集的更多相关文章
- JZOJ 【NOIP2016提高A组集训第16场11.15】兔子
JZOJ [NOIP2016提高A组集训第16场11.15]兔子 题目 Description 在一片草原上有N个兔子窝,每个窝里住着一只兔子,有M条路径连接这些窝.更特殊地是,至多只有一个兔子窝有3 ...
- JZOJ 【NOIP2016提高A组集训第16场11.15】SJR的直线
JZOJ [NOIP2016提高A组集训第16场11.15]SJR的直线 题目 Description Input Output Sample Input 6 0 1 0 -5 3 0 -5 -2 2 ...
- 【NOIP2016提高A组集训第4场11.1】平衡的子集
题目 夏令营有N个人,每个人的力气为M(i).请大家从这N个人中选出若干人,如果这些人可以分成两组且两组力气之和完全相等,则称为一个合法的选法,问有多少种合法的选法? 分析 如果暴力枚举每个人被分到哪 ...
- 【NOIP2016提高A组集训第14场11.12】随机游走
题目 YJC最近在学习图的有关知识.今天,他遇到了这么一个概念:随机游走.随机游走指每次从相邻的点中随机选一个走过去,重复这样的过程若干次.YJC很聪明,他很快就学会了怎么跑随机游走.为了检验自己是不 ...
- 【NOIP2016提高A组集训第13场11.11】最大匹配
题目 mhy12345学习了二分图匹配,二分图是一种特殊的图,其中的点可以分到两个集合中,使得相同的集合中的点两两没有连边. 图的"匹配"是指这个图的一个边集,里面的边两两不存在公 ...
- 【NOIP2016提高A组集训第14场11.12】随机游走——期望+树形DP
好久没有写过题解了--现在感觉以前的题解弱爆了,还有这么多访问量-- 没有考虑别人的感受,没有放描述.代码,题解也写得歪歪扭扭. 并且我要强烈谴责某些写题解的代码不打注释的人,像天书那样,不是写给普通 ...
- 【JZOJ4901】【NOIP2016提高A组集训第18场11.17】矩阵
题目描述 他是一名普通的农电工,他以一颗无私奉献的爱岗敬业之心,刻苦钻研业务,以娴熟的技术.热情周到的服务赢得了广大客户的尊敬和赞美.他就是老百姓称为"李电"的李春来. 众所周知, ...
- 【JZOJ4898】【NOIP2016提高A组集训第17场11.16】人生的价值
题目描述 NiroBC终于找到了人生的意义,可是她已经老了,在新世界,没有人认识她,她孤独地在病榻上回顾着自己平凡的一生,老泪纵横.NiroBC多么渴望再多活一会儿啊! 突然一个戴着黑色方框眼镜,方脸 ...
- 【JZOJ4899】【NOIP2016提高A组集训第17场11.16】雪之国度
题目描述 雪之国度有N座城市,依次编号为1到N,又有M条道路连接了其中的城市,每一条道路都连接了不同的2个城市,任何两座不同的城市之间可能不止一条道路.雪之女王赋予了每一座城市不同的能量,其中第i座城 ...
随机推荐
- Python - 基本数据类型及其常用的方法之元组
元组 特点:一级元素无法被修改,且不能被增加或者删除. 基本操作: tu = (11, 22, ["aiden", 33, ("qwe", 11)], 77) ...
- Latex报错: Could not start the command: xelatex.exe -synctex=1 -interaction=nonstopmode?
Latex报错: Could not start the command: xelatex.exe -synctex=1 -interaction=nonstopmode 网上还有很多说出Could ...
- 初识OpenSSL
首先来认识一下SSL和TLS SSL:Secure Sockets Layer 安全套接层协议 由Netscape公司在1994年发布: TLS:Transport Layer Security 传输 ...
- loj6229 这是一道简单的数学题
https://loj.ac/problem/6229 题解:https://blog.csdn.net/Vectorxj/article/details/79094659 套路推式子,杜教筛,证明复 ...
- iBaties对比hibernate
翻译至一篇2008年的文章(http://www.javaworld.com/article/2077875/open-source-tools/ibatis--hibernate--and-jpa- ...
- PHP搜索优化 sphinx 实战
环境:win7 64 wamp 解压sphinx安装包后,简历如下结构.注意,conf目录是我的配置文件目录 在conf目录下,简历newdefend.conf文件,配置内容如下 # 配置数据源 so ...
- css3 炫酷下拉菜单
<!doctype html> <html> <head> <meta charset="UTF-8"> <title> ...
- img标签下多余空白BUG解决方法
在进行页面的DIV CSS排版时,遇到IE6(当然有时Firefox下也会偶遇)浏览器中的图片元素img下出现多余空白的问题绝对是常见的 对于该问题的解决方法也是“见机行事”. 1.将图片转换为块级对 ...
- matplotlib无法显示中文
import matplotlib as mpl mpl.rcParams['font.sans-serif'] = ['KaiTi']mpl.rcParams['font.serif'] = ['K ...
- P3303 [SDOI2013]淘金
题目描述 小Z在玩一个叫做<淘金者>的游戏.游戏的世界是一个二维坐标.X轴.Y轴坐标范围均为1..N.初始的时候,所有的整数坐标点上均有一块金子,共N*N块. 一阵风吹过,金子的位置发生了 ...