zoukankan      html  css  js  c++  java
  • hdu 2767 Proving Equivalences 强连通缩点

    给出n个命题,m个推导,问最少添加多少条推导。能够使全部命题都能等价(两两都能互推)

    既给出有向图,最少加多少边,使得原图变成强连通。

    首先强连通缩点。对于新图,每一个点都至少要有一条出去的边和一条进来的边(这样才干保证它能到随意点和随意点都能到它)

    所以求出新图中入度为0的个数,和出度为0的个数,加入的边就是从出度为0的指向入度为0的。这样还会有一点剩余。剩余的就乱连即可了。

    所以仅仅要求出2者的最大值就OK。

    #include <iostream>
    #include<cstring>
    #include<cstdio>
    #include<string>
    #include<algorithm>
    using namespace std;
    #define MAXN 30005
    #define MAXM 200005
    struct node
    {
        int to,next;
    }edge[MAXM];
    int head[MAXN],en;
    int low[MAXN],dfn[MAXN],stack[MAXN],top,set[MAXN],col,num;
    bool vis[MAXN],instack[MAXN];
    int in[MAXN],out[MAXN];
    int n;
    int m;
    void addedge(int a,int b)
    {
        edge[en].to=b;
        edge[en].next=head[a];
        head[a]=en++;
    }
    void tarjan(int u)
    {
        vis[u]=1;
        dfn[u]=low[u]=++num;
        instack[u]=true;
        stack[++top]=u;
        for(int i=head[u];i!=-1;i=edge[i].next)
        {
            int v=edge[i].to;
            if(!vis[v])
            {
                tarjan(v);
                low[u]=min(low[u],low[v]);
            }
            else
                if(instack[v])
                    low[u]=min(dfn[v],low[u]);
        }
        if(dfn[u]==low[u])
        {
            int j;
            col++;
            do
            {
                j=stack[top--];
                instack[j]=false;
                set[j]=col;
            }
            while (j!=u);
        }
    }
    void init()
    {
        en=top=col=num=0;
        memset(head,-1,sizeof(head));
        memset(instack,0,sizeof(instack));
        memset(vis,0,sizeof(vis));
        memset(set,-1,sizeof(set));
        memset(in,0,sizeof(in));
        memset(out,0,sizeof(out));
    }
    int main()
    {
        int a,b;
        int cas;
        scanf("%d",&cas);
        while(cas--)
        {
            scanf("%d%d",&n,&m);
            init();
            for(int i=1;i<=m;i++)
            {
                scanf("%d%d",&a,&b);
                addedge(a,b);
            }
            for(int i=1;i<=n;i++)
                if(!vis[i])tarjan(i);
            if(col<=1) {puts("0");continue;}
            int ans=0;
            for(int i=1;i<=n;i++)
                for(int j=head[i];~j;j=edge[j].next)
                {
                    int to=edge[j].to;
                    if(set[to]!=set[i])
                    {
                        in[set[to]]++;
                        out[set[i]]++;
                    }
                }
            int t1=0,t2=0;
            for(int i=1;i<=col;i++)
            {
                if(!in[i]) t1++;
                if(!out[i]) t2++;
            }
            printf("%d
    ",max(t1,t2));
        }
        return 0;
    }
    /*
    3 3
    1 2
    2 1
    1 2
    */
    


  • 相关阅读:
    【LOJ #2320】「清华集训 2017」生成树计数
    【LOJ #2983】「WC2019」数树
    【学习笔记】一类极角排序题
    【学习笔记】斐波那契数列的简单性质
    【LOJ #6041】「雅礼集训 2017 Day7」事情的相似度
    【日常训练】迪杂斯特
    大数据应用技术课程实践--选题与实践方案
    14 深度学习-卷积
    13-垃圾邮件分类2
    12.朴素贝叶斯-垃圾邮件分类
  • 原文地址:https://www.cnblogs.com/jzssuanfa/p/7253488.html
Copyright © 2011-2022 走看看