码迷,mamicode.com
首页 > 其他好文 > 详细

subsets

时间:2018-10-17 00:16:43      阅读:167      评论:0      收藏:0      [点我收藏+]

标签:name   查找   printf   scan   差值   dfs   pac   集合   ==   

一句话题意:有多少个非空子集,能划分成和相等的两份。
题解:对于这道题,我们很轻易可以列出\(O(3^n)\)的暴力,这是显然过不了的,观察这道题的性质可以发现我们显然可以查找左半边把值扔到hash表里,然后查找右半边的时候更新答案,这是显然正确的,因为我们对于hash表维护的的是两个集合的差值,所以不用担心每半边内部的情况会判不到。

#include<cstdio>
#include<algorithm>
using namespace std;
int now,n,h[1000011],nxt[1000011],a[1000011],B[1000011],A[1000011],nm[22],ans;
bool vis[10000011];
const int mod=1000007;
void ins(int x,int y)
{
    int k=abs(x)%mod;
    for(int i=h[k];i;i=nxt[i])
        if(A[i]==x&&B[i]==y)return ;
    ++now;nxt[now]=h[k];h[k]=now;A[now]=x;B[now]=y;
}
void get(int x,int y)
{
    int k=abs(x)%mod;
    for(int i=h[k];i;i=nxt[i])
        if(A[i]==x)vis[y+B[i]]=1;
} 
void dfs2(int x,int y,int z)
{
    if(x==n+1)
    {
        get(y,z);
        return ;
    }
    dfs2(x+1,y+a[x-1],z+nm[x-1]);
    dfs2(x+1,y-a[x-1],z+nm[x-1]);
    dfs2(x+1,y,z);
}
void dfs1(int x,int y,int z)
{
    if(x==n/2+1)
    {
        ins(y,z);
        return ;
    }
    dfs1(x+1,y+a[x-1],z+nm[x-1]);
    dfs1(x+1,y-a[x-1],z+nm[x-1]);
    dfs1(x+1,y,z);
}
int main()
{
    scanf("%d",&n);
    nm[0]=1;
    for(int i=1;i<=21;i++)nm[i]=nm[i-1]<<1;
    for(int i=1;i<=n;i++)scanf("%d",&a[i-1]);
    dfs1(1,0,0);dfs2(n/2+1,0,0);
    for(int i=1;i<=(1<<n);i++)if(vis[i])ans++;
    printf("%d",ans);
}

subsets

标签:name   查找   printf   scan   差值   dfs   pac   集合   ==   

原文地址:https://www.cnblogs.com/lcxer/p/9800808.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!