【小白入門向】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
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...