HDU 2767-Proving Equivalences(強聯通+縮點)

來源:互聯網
上載者:User

標籤:oid   ber   index   const   space   地址   nod   ons   ble   

題目地址:

pid=2767">HDU 2767
題意:給一張有向圖。求最少加幾條邊使這個圖強連通。
思路:先求這張圖的強連通分量。假設為1。則輸出0(證明該圖不須要加邊已經是強連通的了)。否則縮點。

遍曆原圖的全部邊。假設2個點在不同的強連通分量裡面,建邊,構成一張新圖。統計新圖中點的入度和出度,取入度等於0和出度等於0的最大值(由於求強連通縮點後。整張圖就變成了一個無迴路的有向圖。要使之強連通。僅僅須要將入度=0和出度=0的點加邊就可以,要保證加邊後沒有入度和出度為0的點,所以取兩者最大值)
PS:補充一下縮點的含義:我們求強連通分量時,給每一個頂點做一個標記,標記該頂點屬於哪個強聯通分量,然後屬於同一個強連通分量的點就能夠看作同一個點了。

這就是所謂的“縮點”

*#include <stdio.h>#include <math.h>#include <string.h>#include <stdlib.h>#include <iostream>#include <sstream>#include <algorithm>#include <set>#include <queue>#include <stack>#include <map>using namespace std;typedef long long LL;const int inf=0x3f3f3f3f;const double pi= acos(-1.0);const double esp=1e-6;const int maxn=21010;int head[maxn],dfn[maxn],low[maxn],belong[maxn],stak[maxn],instack[maxn];int in[maxn],out[maxn];int incnt,outcnt;int cnt,index,top,ans;struct node {    int u, v, next;} edge[maxn*3];void add(int u, int v) {    edge[cnt].v=v;    edge[cnt].next=head[u];    head[u]=cnt++;}void Init() {    memset(head,-1,sizeof(head));    memset(dfn,0,sizeof(dfn));    memset(instack,0,sizeof(instack));    cnt=index=top=ans=0;    memset(in,0,sizeof(in));    memset(out,0,sizeof(out));    incnt=outcnt=0;}void tarjan(int u) {    dfn[u]=low[u]=++index;    stak[++top]=u;    instack[u]=1;    for(int i=head[u]; i!=-1; i=edge[i].next) {        int v=edge[i].v;        if(!dfn[v]) {            tarjan(v);            low[u]=min(low[u],low[v]);        } else if(instack[v])            low[u]=min(low[u],dfn[v]);    }    if(dfn[u]==low[u]) {        ans++;        while(1) {            int v=stak[top--];            instack[v]=0;            belong[v]=ans;            if(u==v) break;        }    }}int main() {    int T, n, m,i, j;    int u,v;    scanf("%d",&T);    while(T--) {        scanf("%d%d",&n,&m);        Init();        while(m--) {            scanf("%d%d",&u,&v);            add(u,v);        }        for(i=1; i<=n; i++) {            if(!dfn[i])                tarjan(i);        }        if(ans==1) {            printf("0\n");            continue ;        }        for(i=1; i<=n; i++) {            for(j=head[i]; j!=-1; j=edge[j].next) {                int v=edge[j].v;                if(belong[v]!=belong[i]) {                    in[belong[v]]++;                    out[belong[i]]++;                }            }        }        for(i=1; i<=ans; i++) {            if(!in[i])                incnt++;            if(!out[i])                outcnt++;        }        printf("%d\n",max(incnt,outcnt));    }    return 0;}*

HDU 2767-Proving Equivalences(強聯通+縮點)

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

Tags Index: