HDU 5727 Necklace(二分图匹配)

题目分析

根据阴珠子和阳珠子的互斥关系来建立二分图,但是大家发现如果是拿阴和阳来建立二分图很难建立,这里面我们首先对阴珠子进行全排列,然后根据阴珠子所在的位置,找到每一个位置上有哪些阳珠子能放,并且建立边,这样一个二分图就出来,很明显求出二分图的最大匹配,然后用N-二分图的最大匹配就是所求的答案,然后找到这些答案的最小值输出即可。

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
const int maxn = 15;
const int INF = 0x3f3f3f3f;
int a[maxn], head[maxn], vis[maxn], link[maxn], maze[maxn][maxn], tot;
struct Edge{
    int to, next;
}e[maxn*maxn];

void addedge(int from,int to){
    e[tot].to = to;
    e[tot].next = head[from];
    head[from] = tot++;
}

bool dfs(int u){  //匈牙利算法
    for(int i = head[u]; i != -1; i = e[i].next){
        int v = e[i].to;
        if(!vis[v]){
            vis[v] = 1;
            if(link[v] == -1 || dfs(link[v])){
                link[v] = u;
                return true;
            }
        }
    }
    return false;
}

void init(){
    memset(head, -1, sizeof(head));
    memset(vis, 0, sizeof(vis));
    memset(link, -1, sizeof(link));
    tot = 0;
}

int main(){
    int N,M;
    while(scanf("%d%d", &N, &M) != EOF){
        init();
        int from, to;
        memset(maze, 0, sizeof(maze));
        while(M--){
            scanf("%d %d", &from, &to);
            maze[from][to] = 1;
        }
        if(!N){  //0的时候很明显直接跳出即可
            printf("0\n");
            continue;
        }
        for(int i = 1; i <= N; i++) a[i] = i;
        int ans = INF;
        do{
            init();
            for(int i = 1; i <= N; i++){  //i表示第i个位置
                if(i == 1){
                    for(int j = 1; j <= N; j++)  //找到第i个位置能放的珠子
                        if(!maze[j][a[1]] && !maze[j][a[N]]) addedge(i, j);
                }
                else{
                    for(int j = 1; j <= N; j++)
                        if(!maze[j][a[i]] && !maze[j][a[i-1]]) addedge(i, j);
                }
            }
            int cnt = 0;
            for(int i = 1; i <= N; i++){
                memset(vis, 0, sizeof(vis));
                if(dfs(i)) cnt++;
            }
            ans = min(ans, N-cnt);
        }while(next_permutation(a+1, a+1+N)&ans);
        printf("%d\n", ans);
    }
    return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值