hdu 5305 friends

每一次比赛的时候脑子都卡顿,
这次更离谱,我竟然二进制枚举边,这么大的复杂度,并且剪不了枝
后来学长说着是道爆搜,搜每一条边,恍然大悟。
只需要剪掉点的度数是奇数的时候,或者他的线上朋友或线下朋友大于等于度数的1/2时候的枝,
跑了15ms

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;

int x[100],y[100];
int of[100],on[100];

int n,m;

int deg[10];
int ans ;
void dfs(int u){
    if(u==m+1){
        for(int i=1;i<=n;i++){
            if(of[i]!=on[i])
                return;
            ans++;
            return;
        }
    }
    if(on[x[u]]<deg[x[u]]/2&&on[y[u]]<deg[y[u]]/2){
    on[x[u]]++,on[y[u]]++;
    dfs(u+1);
    on[x[u]]--,on[y[u]]--;
    }
    if(of[x[u]]<deg[x[u]]/2&&of[y[u]]<deg[y[u]]/2){
    of[x[u]]++,of[y[u]]++;
    dfs(u+1);
    of[x[u]]--,of[y[u]]--;
    }
}

int main(){
    int T;
    scanf("%d",&T);
    while(T--){
        int flag= 1;
        memset(deg,0,sizeof(deg));
        scanf("%d%d",&n,&m);
        for(int i=1;i<=m;i++){
            scanf("%d%d",&x[i],&y[i]);
            deg[x[i]]++;
            deg[y[i]]++;
        }
        for(int i=1;i<=n;i++){
            if(deg[i]%2){
                printf("0\n");
                flag = 0;
                break;
            }
        }
        if(!flag) continue;
        memset(on,0,sizeof(on));
        memset(of,0,sizeof(of));
        ans = 0;
        dfs(1);
        printf("%d\n",ans);
    }
}

你可能感兴趣的:(暴搜)