Tarjan算法专练

1.迷宫城堡

题意:给一个图判断是否是强连通图。

题解:利用Tarjan计算图中强连通分量的个数,如果为1则是强连通图,否则不是。

#include<bits/stdc++.h>

using namespace std;
const int N = 2e4+100;
typedef long long ll;
vector<int> G[N];
bool is_instack[N];
int dfn[N],low[N];
stack<int> sta;
int n,m,index,scc;
void init(){
    index=scc=0;
    memset(dfn,0,sizeof(dfn));
    memset(low,0,sizeof(low));
    memset(is_instack,0,sizeof(is_instack));
    while(!sta.empty()) sta.pop();
    for(int i=1;i<=n;i++) G[i].clear();
}
void Tarjan(int u){
    dfn[u]=low[u]=++index;
    sta.push(u);is_instack[u]=1;
    for(auto v:G[u]){
        if(!dfn[v]){
            Tarjan(v);
            low[u]=min(low[u],low[v]);
        }
        else if(is_instack[v]){
            low[u]=min(low[u],dfn[v]);
        }
    }
    if(low[u]==dfn[u]){
        ++scc;
        while(sta.top()!=u){
            is_instack[sta.top()]=0;
            sta.pop();
        }
    }
}
int main(){
    while(scanf("%d %d",&n,&m)){
        if(n+m==0) break;
        init();
        int u,v;
        for(int i=0;i<m;i++){
            scanf("%d %d",&u,&v);
            G[u].push_back(v);
        }
        for(int i=1;i<=n;i++){
            if(!dfn[i]) Tarjan(i);
        }
        if(scc==1) puts("Yes");
        else puts("No");
    }
    return 0;
}

2.Proving Equivalences

题意:给定一个有向图,求最少加几条有向边使得整个图成为强连通。

题解:Tarjan缩点之后计算入读为\(0\)的个数\(a\)和出度为\(0\)的个数\(b\),取最大值\(max(a,b)\),注意如果已经是强连通图了,则答案为\(0\).

#include<bits/stdc++.h>

using namespace std;
const int N = 1e5+100;
vector<int> G[N],color[N];
stack<int> sta;
int dfn[N],low[N];
int id[N],od[N];
bool is_instack[N];
int scc[N],nscc,index;
int n,m;
void init(){
    nscc=index=0;
    memset(scc,0,sizeof(scc));
    memset(dfn,0,sizeof(dfn));
    memset(id,0,sizeof(id));
    memset(od,0,sizeof(od));
    memset(is_instack,0,sizeof(is_instack));
    memset(low,0,sizeof(low));
    for(int i=1;i<=n;i++) G[i].clear(),color[i].clear();
    while(!sta.empty()) sta.pop();
}
void Tarjan(int u){
    low[u]=dfn[u]=++index;
    sta.push(u);is_instack[u]=1;
    for(auto v:G[u]){
        if(!dfn[v]){
            Tarjan(v);
            low[u]=min(low[u],low[v]);
        }
        else if(is_instack[v]){
            low[u]=min(low[u],dfn[v]);
        }
    }
    if(low[u]==dfn[u]){
        ++nscc;
        while(1){
            int temp=sta.top();
            scc[temp]=nscc;
            sta.pop();
            is_instack[temp]=0;
            if(temp==u) break;
        }
    }
}
int main(){
    int T;
    scanf("%d",&T);
    while(T--){
        scanf("%d %d",&n,&m);
        init();
        int u,v;
        for(int i=1;i<=m;i++){
            scanf("%d %d",&u,&v);
            G[u].push_back(v);
        }
        for(int i=1;i<=n;i++){
            if(!dfn[i]) Tarjan(i);
        }
        //cerr<<nscc<<endl;
        if(nscc==1){
            puts("0");
        }
        else{
            for(int i=1;i<=nscc;i++) id[i]=od[i]=1;
            for(int i=1;i<=n;i++){
                for(auto j:G[i]){
                    if(scc[i]!=scc[j]){
                        id[scc[j]]=od[scc[i]]=0;
                    }
                }
            }
            int n1=0;int n2=0;
            for(int i=1;i<=nscc;i++){
                n1+=id[i];n2+=od[i];
            }
            printf("%d\n",max(n1,n2));
        }
    }
    return 0;
}

3.Summer Holiday

题意:给你一个有向图,每个点都有一个权值,你需要在这个图中选择几个点使得利用这几个点可以遍历完整个图,同时还要满足选择的这几个点的权值和最小。

题解:Tarjan缩点,然后在缩点的时候把点的权值缩成所在的强连通分量里面权值最小的,然后寻找入度为0的点的个数,即为答案。
{% fold 点击显/隐内容 %}

#include<bits/stdc++.h>

using namespace std;
const int N = 1e5+100;
typedef long long ll;
const int INF = 0x3f3f3f3f;
int n,m,scc,index;
vector<int> G[N];
ll w[N],low[N],dfn[N],minn[N],color[N],id[N];
bool is_instack[N];stack<int> sta;
void init(){
    scc=index=0;
    memset(low,0,sizeof(low));
    memset(dfn,0,sizeof(dfn));
    memset(color,0,sizeof(color));
    memset(minn,INF,sizeof(minn));
    memset(is_instack,0,sizeof(is_instack));
    for(int i=1;i<=n;i++) G[i].clear();
    while(!sta.empty()) sta.pop();
}
void Tarjan(int u){
    low[u]=dfn[u]=++index;
    sta.push(u);is_instack[u]=1;
    for(auto v:G[u]){
        if(!dfn[v]){
            Tarjan(v);
            low[u]=min(low[u],low[v]);
        }
        else if(is_instack[v]){
            low[u]=min(low[u],dfn[v]);
        }
    }
    if(low[u]==dfn[u]){
        ++scc;
        while(1){
            int temp=sta.top();
            color[temp]=scc;
            minn[scc]=min(minn[scc],w[temp]);
            is_instack[temp]=0;
            sta.pop();
            if(temp==u) break;
        }
    }
}
int main(){
    while(scanf("%d %d",&n,&m)!=EOF){
        init();
        for(int i=1;i<=n;i++) scanf("%lld",&w[i]);
        int u,v;
        for(int i=1;i<=m;i++){
            scanf("%d %d",&u,&v);
            G[u].push_back(v);
        }
        for(int i=1;i<=n;i++){
            if(!dfn[i]) Tarjan(i);
        }
        for(int i=1;i<=scc;i++) id[i]=1;
        for(int i=1;i<=n;i++){
            for(auto j:G[i]){
                if(color[i]!=color[j]){
                    id[color[j]]=0;
                }
            }
        }
        ll ans1,ans2;ans1=ans2=0;
        for(int i=1;i<=scc;i++){
            if(id[i]){
                ans1++;
                ans2+=minn[i];
            }
        }
        printf("%lld %lld\n",ans1,ans2);
    }
    return 0;
}

{% endfold %}

4.Intelligence System

题意:一个人际关系网,0号可以联系上任意一个人,如果两个人可以直接或间接的互相联系,那么这两个人的消费为0,求0号要联系上每个人最小的消费。

题解: Tarjan缩点,求最小树形图,由于题目保证有解,因此只需要统计每个点入边权值最小的即可。0所在的强连通分量不用考虑。

#include<bits/stdc++.h>

using namespace std;
const int N = 50005 ;
typedef long long ll;
const ll INF = 0x3f3f3f3f;
ll low[N],dfn[N],minn[N],color[N];
bool  is_instack[N];
ll n,m,scc,index;
stack<ll> sta;
vector<pair<ll,ll> > G[N];
void init(){
    scc=index=0;
    for(int i=0;i<=n;i++){
        is_instack[i]=0;
        low[i]=0;
        dfn[i]=0;
        minn[i]=INF;
        G[i].clear();
    }
    while(!sta.empty()) sta.pop();
}
void Tarjan(ll u){
    low[u]=dfn[u]=++index;
    is_instack[u]=1;sta.push(u);
    for(auto V:G[u]){
        ll v=V.first;
        if(!dfn[v]){
            Tarjan(v);
            low[u]=min(low[u],low[v]);
        }
        else if(is_instack[v]){
            low[u]=min(low[u],dfn[v]);
        }
    }
    if(low[u]==dfn[u]){
        scc++;
        while(1){
            ll temp=sta.top();
            color[temp]=scc;
            is_instack[temp]=0;
            sta.pop();
            if(temp==u) break;
        }
    }
}
int main(){
    while(scanf("%lld %lld",&n,&m)!=EOF){
        map<pair<ll,ll>,ll> WW;
        //pair<ll,ll> pnow;
        init();
        ll u,v,w;
        for(ll i=1;i<=m;i++){
            scanf("%lld %lld %lld",&u,&v,&w);
            //pnow.first=u;pnow.second=v;
            //if(!WW[pnow]) WW[pnow]=w;
            //else WW[pnow]=min(WW[pnow],w);
            G[u].push_back(make_pair(v,w));
        }
        long long ans=0;
        for(ll i=0;i<n;i++){
            if(!dfn[i]) Tarjan(i);
        }
        for(ll i=0;i<n;i++){
            for(auto j:G[i]){
                ll x=color[i];
                ll y=color[j.first];
                if(x!=y) minn[y]=min(minn[y],(ll)j.second);
            }
        }
        for(ll i=1;i<=scc;i++){
            if(i!=color[0]) ans+=minn[i];
        }
        printf("%lld\n",ans);
    }
    return 0;
}

原文地址:https://www.cnblogs.com/codancer/p/12232291.html

时间: 2024-10-14 21:46:18

Tarjan算法专练的相关文章

有向图强连通分支的Tarjan算法讲解 + HDU 1269 连通图 Tarjan 结题报告

题目很简单就拿着这道题简单说说 有向图强连通分支的Tarjan算法 有向图强连通分支的Tarjan算法伪代码如下:void Tarjan(u) {dfn[u]=low[u]=++index//进行DFS,每发现一个新的点就对这个点打上时间戳,所以先找到的点时间戳越早,dfn[U]表示最早发现u的时间,low[u]表示u能到达的最早的时间戳.stack.push(u)//将U压入栈中for each (u, v) in E {if (v is not visted)//如果V点没有经历过DFS,则

【转】BYV--有向图强连通分量的Tarjan算法

转自beyond the void 的博客: https://www.byvoid.com/zhs/blog/scc-tarjan 注:红色为标注部分 [有向图强连通分量] 在有向图G中,如果两个顶点间至少存在一条路径,称两个顶点强连通(strongly connected).如果有向图G的每两个顶点都强连通,称G是一个强连通图.非强连通图有向图的极大强连通子图,称为强连通分量(strongly connected components). 下图中,子图{1,2,3,4}为一个强连通分量,因为顶

(转载)LCA问题的Tarjan算法

转载自:Click Here LCA问题(Lowest Common Ancestors,最近公共祖先问题),是指给定一棵有根树T,给出若干个查询LCA(u, v)(通常查询数量较大),每次求树T中两个顶点u和v的最近公共祖先,即找一个节点,同时是u和v的祖先,并且深度尽可能大(尽可能远离树根).LCA问题有很多解法:线段树.Tarjan算法.跳表.RMQ与LCA互相转化等.本文主要讲解Tarjan算法的原理及详细实现. 一 LCA问题 LCA问题的一般形式:给定一棵有根树,给出若干个查询,每个

【强连通分量】tarjan算法及kosaraju算法+例题

阅读前请确保自己知道强连通分量是什么,本文不做赘述. Tarjan算法 一.算法简介 Tarjan算法是一种由Robert Tarjan提出的求有向图强连通分量的时间复杂度为O(n)的算法. 首先我们要知道两个概念:时间戳(DFN),节点能追溯到的最早的栈中节点的时间戳(LOW).顾名思义,DFN就是在搜索中某一节点被遍历到的次序号(dfs_num),LOW就是某一节点在栈中能追溯到的最早的父亲节点的搜索次序号. Tarjan算法是基于深度优先搜索的算法.在搜索过程中把没有Tarjan过的点入栈

有向图的强连通分量(tarjan算法)

强连通分量 有向图强连通分量:在有向图G中,如果两个顶点vi,vj间(vi>vj)有一条从vi到vj的有向路径,同时还有一条从vj到vi的有向路径,则称两个顶点强连通(strongly connected).如果有向图G的每两个顶点都强连通,称G是一个强连通图.有向图的极大强连通子图,称为强连通分量(strongly connected components). 考虑强连通分量C,设其中第一个被发现的点为x,则,C中其他的点都是x的后代.我们希望在x访问完成时立即输出C(可以同时记录C,输出代表

图的连通性:强连通-Tarjan算法

不早了,先上个模板,做些题再来分析一下: 强连通Tarjan算法+前向星 模板如下: const int MAXN=110; const int MAXM=10010; struct edge { int to,next; }; edge E[MAXN]; int head[MAXN],Ecou; int Stack[MAXN],top; int Belong[MAXN],lfcou; int Index; int DFN[MAXN],LOW[MAXN]; bool inStack[MAXN];

POJ1523(求连用分量数目,tarjan算法原理理解)

SPF Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 7406   Accepted: 3363 Description Consider the two networks shown below. Assuming that data moves around these networks only between directly connected nodes on a peer-to-peer basis, a

tarjan算法模板

var {left表示点 root 没离开栈 vis表示点 root 有没有被访问过} i,n,m,now,time,color,top:longint; v:array[0..10001] of record start:longint;end; e:array[0..100001] of record y,next:longint;end; dfn,low,stack,encolor:array[0..10001] of longint; vis,left:array[0..10001] o

hdu1269 迷宫城堡,有向图的强连通分量 , Tarjan算法

hdu1269 迷宫城堡 验证给出的有向图是不是强连通图... Tarjan算法板子题 Tarjan算法的基础是DFS,对于每个节点.每条边都搜索一次,时间复杂度为O(V+E). 算法步骤: 1.搜索到某一个点时,将该点的Low值标上时间戳,然后将自己作为所在强连通分量的根节点(就是赋值Dfn=Low=time) 2.将该点压入栈. 3.当点p有与点p'相连时,如果此时p'不在栈中,p的low值为两点的low值中较小的一个. 4.当点p有与点p'相连时,如果此时p'在栈中,p的low值为p的lo