【小白入門向】tarjan演算法+codevs1332題解報告

来源:http://www.cnblogs.com/Maki-Nishikino/archive/2016/09/12/5866191.html
-Advertisement-
Play Games

一、【前言】關於tarjan tarjan演算法是由Robert Tarjan提出的求解有向圖強連通分量的演算法。 那麼問題來了找藍翔!(劃掉)什麼是強連通分量? 我們定義:如果兩個頂點互相連通(即存在A到B和B到A的通路),則稱這兩個點強連通。對於一個有向圖G,若是G中任意兩點都強連通,則稱G是一個強 ...


一、【前言】關於tarjan

tarjan演算法是由Robert Tarjan提出的求解有向圖強連通分量的演算法。

那麼問題來了找藍翔!(劃掉)什麼是強連通分量?

我們定義:如果兩個頂點互相連通(即存在A到B和B到A的通路),則稱這兩個點強連通。對於一個有向圖G,若是G中任意兩點都強連通,則稱G是一個強連通圖。有向圖的極大強連通子圖,稱為該圖的強連通分量

對於下圖,{1,2,3,4}、{5}、{6}分別是它的強連通分量。

那麼tarjan是如何找到這些強連通分量的呢?

說白了tarjan就是dfs,每個強連通分量都是搜索樹中的一顆子樹。搜索時,我們把當前搜索樹中未處理過的點加入一個堆棧,回溯時從棧頂依次取出元素判斷他們是否屬於一個強連通分量。

我們對dfs的過程中添加如下定義:
1、dfn[i]:代表搜索點i的次序編號;

2、low[i]:代表點i所能回溯到的棧中最早的次序編號。

當dfn[i]=low[i]時,以i為根的子樹上的所有點便構成一個強連通分量。

 

二、tarjan演算法c語言實現

那麼要如何使用程式來實現tarjan演算法呢?

大致思路如下:
1、從一個未被處理過的點i開始,給它結合一個次序編號並把它壓入棧中,然後標記點表示其已入棧,令low[i]=dfn[i]=次序編號;

2、遍歷每條以i為起點的邊,若邊的終點j未被處理過,就對其進行操作1~3,令low[i]=min(low[i],low[j]);

3、找到的點j若是被處理過,則判斷其是否在棧中:若在,則令low[i]=min(low[i],dfn[j]);

4、若是dfn[i]=low[i],就將棧頂到i間的所有元素彈出,它們便是一個強連通分量;

5、重覆1~4,直至不存在點未被處理。

貼上偽代碼(轉自百度百科詞條——tarjan演算法)

 1 tarjan(u)
 2 {
 3     DFN[u]=Low[u]=++Index//為節點u設定次序編號和Low初值
 4     Stack.push(u)//將節點u壓入棧中
 5     for each(u,v) in E//枚舉每一條邊
 6         if (visnotvisted)//如果節點v未被訪問過
 7             tarjan(v)//繼續向下找
 8             Low[u]=min(Low[u],Low[v])
 9         else if (vinS)//如果節點v還在棧內
10                 Low[u]=min(Low[u],DFN[v])
11     if (DFN[u]==Low[u])//如果節點u是強連通分量的根
12     repeat{
13         v=S.pop//將v退棧,為該強連通分量中一個頂點
14         printv
15         until(u==v)
16     }
tarjan偽代碼

三、codevs1332題解

http://codevs.cn/problem/1332/←_←掛上原題鏈接

這是一道全♂裸的tarjan題,我們只需跑一遍tarjan,並對所有強連通分量染色,最後輸出最大的就好

代碼如下:

 1 #include<stdio.h>
 2 #include<algorithm>
 3 using namespace std;
 4 struct node
 5 {
 6     int v;
 7     int next;
 8 }e[50010];//鄰接表記錄有向圖 
 9 int stack[5010],top;//
10 int dfn[5010],low[5010],index;//index用於記錄次序編號 
11 bool vis[5010];//判斷點是否在棧內 
12 int st[5010],cnt; 
13 int color[5010],s[5010],num;//用於染色並記錄顏色種類 
14 void build(int a,int b)
15 {
16     e[++cnt].v=b;
17     e[cnt].next=st[a];
18     st[a]=cnt;
19 }//建圖,沒什麼好說的 
20 void tarjan(int x)
21 {
22     dfn[x]=++index;
23     low[x]=index;
24     vis[x]=true;
25     stack[++top]=x;//當前點入棧 
26     int i;
27     for(i=st[x];i!=0;i=e[i].next)//枚舉以當前點為起點的邊 
28     {
29         int temp=e[i].v;//temp為當前被枚舉邊的終點 
30         if(!dfn[temp])//如果當前邊終點未被處理 
31         {
32             tarjan(temp);
33             low[x]=min(low[x],low[temp]);
34         }
35         else if(vis[temp])low[x]=min(low[x],dfn[temp]);
36     }
37     if(dfn[x]==low[x])
38     {
39         vis[x]=false;
40         color[x]=++num;//給當前強連通分量染上新顏色 
41         s[num]++;//給當前強連通分量里的點染色 
42         while(stack[top]!=x)//棧頂元素依次出棧 
43         {
44             s[num]++;
45             color[stack[top]]=num;
46             vis[stack[top--]]=false;
47         }
48         top--;
49     }
50 }
51 int main()
52 {
53     int n,m,i,a,b,t,ans=0,f;
54     scanf("%d%d",&n,&m);
55     for(i=1;i<=m;i++)
56     {
57         scanf("%d%d%d",&a,&b,&t);
58         build(a,b);
59         if(t-1)build(b,a);
60     }
61     for(i=1;i<=n;i++)
62     {
63         if(!dfn[i])tarjan(i);
64     }
65     for(i=1;i<=n;i++)
66     {
67         if(s[color[i]]>ans)//找到被染色最多的強連通分量(因為要求字典序,所以使用'>') 
68         {
69             ans=s[color[i]];
70             f=i;
71         }
72     }
73     printf("%d\n",ans);
74     for(i=1;i<=n;i++)
75     {
76         if(color[i]==color[f])printf("%d ",i);//輸出被染成最大強連通分量的顏色的點 
77     }
78     return 0;
79 }
CodeVs1332
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Given a sorted linked list, delete all nodes that have duplicate numbers, leaving only distinct numbers from the original list. For example,Given 1->2 ...
  • 這篇文章來介紹一下spfa(Shortest Path Faster Algorithm)這種演算法 這是一種單源最短路的一種十分高效的的演算法。 我們需要用鄰接表來存儲一下圖,以及用隊列進行優化。 我們以1為起點,以n為終點來講一下(一共n個點) 用L數組記錄當前點的最短路 先把每一條邊的最短路賦成最 ...
  • 這題是在01背包問題的基礎上,擴充了重量,需要用時間換空間。要註意重量wi為a*2^b(a<=10,b<=30),很容易想到要按 b 分開做背包的DP。重點是怎麼從b-1繼承到b——仔細看題,發現只有一次詢問,那麼就可以在這個W上做文章:依W的大小進行所有的DP。由於是按b的大小進行DP,所以把數都 ...
  • 最近由於在做一個眾籌的項目,其中有一個查找項目支持數的介面,查找的方法定義的是一個long型的,我更新項目中的支持數的時候是int型的,所以需要在long型與int型之間轉化,下麵把轉轉化的詳細方法記錄一下,以便後面總結學習: 一.將long型轉化為int型,這裡的long型是基礎類型: long ...
  • 這個,不管是什麼書都會這樣說,因為常常我們並不需要繼承,而只是想把類進行一定的擴展,而我們想擴展的屬性或方法對應的類都有,這個時候如果兩者是is a的關係,這種關係是確實存在的,那麼就可以使用繼承,不然一般都是建議使用複合。 如果我們隊一個類進行繼承的時候,我們如果對其內部的邏輯並不十分瞭解的時候, ...
  • 獲取類的名稱 獲取該類的方法 獲取方法的返回值類型 獲取方法的名稱 獲取方法的參數的類型 返回結果 ...
  • 建議26:提防包裝類型的null值 我們知道Java引入包裝類型(Wrapper Types)是為瞭解決基本類型的實例化問題,以便讓一個基本類型也能參與到面向對象的編程世界中。而在Java5中泛型更是對基本類型說了"不",如果把一個整型放入List中,就必須使用Integer包裝類型。我們看一段代碼 ...
  • 合成存取方法 Objective-C從 OC 2.0版本開始,自動合成了setter 方法和 getter 方法。而且,如果開發者需要自己控制某個setter 方法和 getter 方法的實現時,可以自己提供 setter 方法和 getter 方法,自己提供的setter 方法和 getter 方 ...
一周排行
    -Advertisement-
    Play Games
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...