zoukankan      html  css  js  c++  java
  • subsets(2018.10.16)

    一句话题意:给你一个包含n个元素的集合,问有多少个非空子集,能划分成和相等的两份。(n<=20)
    题解:对于这道题,我们很轻易可以列出(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);
    }
    
  • 相关阅读:
    linux基础(一)
    网络基础之网络协议篇
    操作系统简介
    计算机组成原理
    【C#】=>符号的使用
    【Unity3D】用继承EditorUpdater类来实现Editor模式下的后台处理
    【Unity3D】Tags和Layers
    【Unity3D】Unity3D中Material与ShareMaterial引用的区别
    【Unity3D】Unity中用C#读取CSV文件
    【Unity3D】用C#读取INI配置文件
  • 原文地址:https://www.cnblogs.com/lcxer/p/9800808.html
Copyright © 2011-2022 走看看