zoukankan      html  css  js  c++  java
  • I

    题意:用最小的点来覆盖全部的边,因为二分图里面最大的匹配就是最小覆盖,所以直接匹配一下即可
    ***********************************************************************
    #include<stdio.h>
    #include<string.h>
    #include<queue>
    using namespace std;

    const int MAXN = 1505;
    const int oo = 1e9+7;

    struct Edge{int v, next;}e[MAXN*4];///数组开小RE了一次
    int Head[MAXN], cnt;

    int Mx[MAXN], My[MAXN];
    int dx[MAXN], dy[MAXN];
    int used[MAXN], N, depth;

    void InIt()
    {
        cnt = 0;
        memset(Head, -1sizeof(Head));
        memset(Mx, -1sizeof(Mx));
        memset(My, -1sizeof(My));
    }
    void AddEdge(int u, int v)
    {
        e[cnt].v = v;
        e[cnt].next = Head[u];
        Head[u] = cnt++;
    }
    bool BFS()
    {
        queue<int> Q;
        depth = oo;

        memset(dx, falsesizeof(dx));
        memset(dy, falsesizeof(dy));

        for(int i=0; i<N; i++)
        {
            if( Mx[i] == -1 )
            {
                dx[i] = true;
                Q.push(i);
            }
        }

        while(Q.size())
        {
            int u = Q.front();Q.pop();

            if(dx[u] > depth)break;

            for(int j=Head[u]; j!=-1; j=e[j].next)
            {
                int v = e[j].v;
                if( dy[v] == false )
                {
                    dy[v] = dx[u] + 1;
                    if(My[v] == -1)
                        depth = dy[v];
                    else
                    {
                        dx[ My[v] ] = dy[v] + 1;
                        Q.push( My[v] );
                    }
                }
            }
        }

        return depth != oo;
    }
    bool DFS(int i)
    {
        for(int j=Head[i]; j!=-1; j=e[j].next)
        {
            int v = e[j].v;
            if( used[v] == false && dx[i] == dy[v]-1 )
            {
                used[v] = true;
                if(My[v] != -1 && dy[v] == depth)
                    continue;
                if(My[v] == -1 || DFS(My[v]))
                {
                    My[v] = i;
                    Mx[i] = v;

                    return true;
                }
            }
        }

        return false;
    }
    int Karp()
    {
        int ans = 0;

        while( BFS() == true)
        {
            memset(used, falsesizeof(used));
            for(int i=0; i<N; i++)
            {
                if( Mx[i] == -1 && DFS(i) )
                    ans ++;
            }
        }

        return ans;
    }

    int main()
    {
        while(scanf("%d", &N) != EOF)
        {
            int M, i, u, v;

            InIt();

            for(i=0; i<N; i++)
            {
                scanf("%d:(%d)", &u, &M);
                while(M--)
                {
                    scanf("%d", &v);
                    AddEdge(u, v);
                    AddEdge(v, u);
                }
            }

            int ans = Karp();

            printf("%d ", ans/2);
        }

        return 0;
    }
  • 相关阅读:
    判断鼠标在按钮区域上面
    在MFC下绘制直线,使用橡皮筋技术,可以使直线效果跟随鼠标移
    三缓冲
    MFC--自己优化滚动条的双缓冲绘图方法
    MFC视图切换大全总结
    各种线程:事件、互斥量、信号量、临界区 的用法,我自己做的,有用,附件里面有,博客附件里面有
    http://www.cctry.com/forum.php?mod=viewthread&tid=800&reltid=4131&pre_thread_id=0&pre_pos=3&ext=
    关于Mac下pycharm无法调用摄像头权限的问题
    终于理解清楚attention,利用attention对黄金价格进行预测
    tensorboard在colab中的实现
  • 原文地址:https://www.cnblogs.com/liuxin13/p/4699791.html
Copyright © 2011-2022 走看看