習題:codevs 2822 愛在心中 解題報告

来源:http://www.cnblogs.com/gangding/archive/2016/07/10/5657329.html
-Advertisement-
Play Games

這次的解題報告是有關tarjan演算法的一道思維量比較大的題目(真的是原創文章,希望管理員不要再把文章移出首頁)。 這道題蒟蒻以前做過,但是今天由於要複習tarjan演算法,於是就看到codevs分類強聯通分量裡面只有這一道題。 題目是這樣的: 這是一個有向圖上的問題,這道題很容易看出來一個愛心天使就是 ...


這次的解題報告是有關tarjan演算法的一道思維量比較大的題目(真的是原創文章,希望管理員不要再把文章移出首頁)。

這道題蒟蒻以前做過,但是今天由於要複習tarjan演算法,於是就看到codevs分類強聯通分量裡面只有這一道題。

題目是這樣的:

“每個人都擁有一個夢,即使彼此不相同,能夠與你分享,無論失敗成功都會感動。愛因為在心中,平凡而不平庸,世界就像迷宮,卻又讓我們此刻相逢Our Home。”

在愛的國度里有N個人,在他們的心中都有著一個愛的名單,上面記載著他所愛的人(不會出現自愛的情況)。愛是具有傳遞性的,即如果A愛B,B愛C,則A也愛C。
如果有這樣一部分人,他們彼此都相愛,則他們就超越了一切的限制,用集體的愛化身成為一個愛心天使。
現在,我們想知道在這個愛的國度里會出現多少愛心天使。而且,如果某個愛心天使被其他所有人或愛心天使所愛則請輸出這個愛心天使是由哪些人構成的,否則輸出-1

這是一個有向圖上的問題,這道題很容易看出來一個愛心天使就是一群人互相愛然後組成的有向環。既然是有向的圖求強聯通分量,套上tarjan就行了。不過因為後面要輸出愛心天使是由哪些人構成的,所以,我們需要記錄每個人在哪個愛心天使裡面,具體tarjan標程就是下麵這樣:

 1 void dfs(int u){
 2     low[u] = dfn[u] = ++dfs_clock;
 3     s.push(u);
 4     for(int i = 0;i < g[u].size();++i){
 5         int v = g[u][i];
 6         if(!dfn[v]){
 7             dfs(v);
 8             low[u] = min(low[u],low[v]);
 9         }
10         else if(!ind[v]){
11             low[u] = min(low[u],dfn[v]);
12         }
13     }
14     if(low[u] == dfn[u]){
15         scc_cnt++;
16         while(1){
17             int x = s.top();
18             s.pop();
19             ind[x] = scc_cnt;
20             sccno[scc_cnt]++;
21             if(x == u)break;
22         }
23     }
24 }
25 void find_scc(int n){
26     dfs_clock = scc_cnt = 0;
27     for(int i = 1;i <= n;++i){
28         if(!dfn[i])dfs(i);
29     }
30 }

如果有不懂的地方,就說明是tarjan演算法還是沒搞懂,請先搞定tarjan演算法的基礎再來看這道題。

然後就是求每個愛心天使被多少個人愛著(愛心天使也被屬於它自己的人們愛著)。由於愛有傳遞性,我們很容易能想到傳遞閉包的問題。具體做法就是:

1.先求出來所有的愛心天使(在這裡一個人也當作是一個愛心天使,但是輸出愛心天使個數的時候,一個人組成的愛心天使是不算的)和這個愛心天使由多少個人組成。

2.給每個愛心天使編號(縮點),然後建立一個以“被愛”為方向的新的有向圖。

3.在新的有向圖中跑SPFA(為什麼不用Floyd,這個時間複雜度太高了,所以我們還是選擇scc_cnt遍的SPFA,時間複雜度最壞也就是O(nke),保證是超不了時間的。SPFA在這裡起到的作用就是計算每個愛心天使的愛能否傳遞到某個天使)。

4.統計計數,如果size[某個愛心天使] == n的話,那麼就按編號從小到大遍歷每個人輸出這個愛心天使由哪些人組成。如果沒有任何一個愛心天使的size為n的話,(用一個數記錄,如果b == 0)就輸出-1.

然後呈現整體代碼,還是不是很長,才剛139行。

  1 #include <cstdio>
  2 #include <cstring>
  3 #include <iostream>
  4 #include <algorithm>
  5 #include <cmath>
  6 #include <stack>
  7 #include <vector>
  8 #include <set>
  9 #include <queue>
 10 using namespace std;
 11 const int maxn = 1005;
 12 int dfn[maxn],low[maxn],dfs_clock,n,ind[maxn],m,scc_cnt,a,b,size[maxn],sccno[maxn],tim = 0;
 13 struct edge{
 14     int u,v;
 15     edge(int a,int b):u(a),v(b){};
 16 };
 17 struct edges{
 18     int to,next,cost;
 19 }qmap[maxn<<3];
 20 vector<int>g[maxn];
 21 vector<int>gk[maxn];
 22 vector<edge>ga;
 23 int d[maxn],h[maxn];
 24 const int INF = 10000000;
 25 stack<int> s;
 26 void add(int u,int v){
 27     qmap[tim].to = v;qmap[tim].cost = 1;qmap[tim].next = h[u];h[u] = tim++;
 28 }
 29 void init(int n){
 30     memset(dfn,0,sizeof(dfn));
 31     memset(low,0,sizeof(low));
 32     memset(ind,0,sizeof(ind));
 33     memset(size,0,sizeof(size));
 34     memset(sccno,0,sizeof(sccno));
 35     memset(h,-1,sizeof(h));
 36     for(int i = 1;i <= n;++i){
 37         g[i].clear();
 38         gk[i].clear();
 39     }
 40 }
 41 void dfs(int u){
 42     low[u] = dfn[u] = ++dfs_clock;
 43     s.push(u);
 44     for(int i = 0;i < g[u].size();++i){
 45         int v = g[u][i];
 46         if(!dfn[v]){
 47             dfs(v);
 48             low[u] = min(low[u],low[v]);
 49         }
 50         else if(!ind[v]){
 51             low[u] = min(low[u],dfn[v]);
 52         }
 53     }
 54     if(low[u] == dfn[u]){
 55         scc_cnt++;
 56         while(1){
 57             int x = s.top();
 58             s.pop();
 59             ind[x] = scc_cnt;
 60             sccno[scc_cnt]++;
 61             if(x == u)break;
 62         }
 63     }
 64 }
 65 void find_scc(int n){
 66     dfs_clock = scc_cnt = 0;
 67     for(int i = 1;i <= n;++i){
 68         if(!dfn[i])dfs(i);
 69     }
 70 }
 71 void spfa(int x){
 72     for(int i = 1;i <= scc_cnt;++i){
 73         d[i] = INF;
 74     }
 75     bool visit[maxn];
 76     memset(visit,false,sizeof(visit));
 77     d[x] = 0;
 78     queue<int>q;
 79     q.push(x);
 80     visit[x] = true;
 81     while(!q.empty()){
 82         int y = q.front();
 83         q.pop();visit[y] = false;
 84         for(int i = h[y];i != -1;i = qmap[i].next){
 85             edges e = qmap[i];
 86             if(d[y] + e.cost < d[e.to]){
 87                 d[e.to] = d[y] + e.cost;
 88                 if(!visit[e.to]){
 89                     q.push(e.to);
 90                     visit[e.to] = true;
 91                 }
 92             }
 93         }
 94     }
 95 }
 96 void work(int i){
 97     spfa(i);
 98     for(int k = 1;k <= scc_cnt;++k){
 99         if(k == i)continue;
100         if(d[k] == INF)continue;
101         size[i] += sccno[k];
102     }
103     if(size[i] == n){
104         b = 1;
105         for(int j = 1;j <= n;++j){
106             if(ind[j] == i)printf("%d ",j);
107         }
108         printf("\n");
109     }
110     return;
111 }
112 int main(){
113     scanf("%d%d",&n,&m);
114     init(n);
115     for(int i = 1;i <= m;++i){
116         scanf("%d%d",&a,&b);
117         g[a].push_back(b);
118         gk[b].push_back(a);
119         ga.push_back(edge(b,a)); 
120     }
121     find_scc(n);
122     int ans = scc_cnt;
123     for(int i = 1;i <= scc_cnt;++i){
124         if(sccno[i] == 1)ans--;
125     }
126     printf("%d\n",ans);
127     for(int i = 0;i < ga.size();++i){
128         add(ind[ga[i].u],ind[ga[i].v]);
129     }
130     for(int i = 1;i <= scc_cnt;++i){
131         size[i] = sccno[i];
132     }
133     b = 0;
134     for(int i = 1;i <= scc_cnt;++i){
135         if(sccno[i] != 1)work(i);
136     }
137     if(b == 0)printf("-1\n");
138     return 0;
139 }

這次的解題報告就到這裡,蒟蒻繼續去刷題了。還有如果有問題的話,請聯繫我的郵箱[email protected]向我留言。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 0. 目錄 C#6 新增特性目錄 1. 老版本的代碼 通常情況下,有些簡單的只讀屬性和方法只有一行代碼,但是我們也不得不按照繁瑣的語法去實現它。C#6帶了了一種和lambda語法高度一致的精簡語法來幫助我們簡化這些語法。先看看老版本的IL代碼(這裡我就不展開IL了,看下結構即可,都是普通的屬性和方法 ...
  • 下一篇:ASP.NET Core(二):入門 英文原版:Introduction to ASP.NET Core 關於ASP.NET Core ASP.NET Core 是一個全新的開源、跨平臺框架,可以用它來構建基於網路連接的現代雲應用程式,比如:Web 應用,IoT(Internet Of Th ...
  • 一直想做電商軟體,但是實在不想學PHP了,所以前後關註了這兩個開源電商系統。一個是國人出品的,一個據說是俄羅斯人寫得(不知道對不對)。目前兩個開源軟體都在學習瞭解中,以下的博文可能會涉及到這兩套系統,我希望能對比進行學習,能互相借鑒和補充。 brnshop :http://www.cnblogs.c ...
  • .net core最近園子討論頻率很高的話題,從不久前發佈正式版本後,也是開始從netcore官網一步一步走向學習之路;.net跨平臺的設計讓人很是興奮起來,因為做了多年的互聯網研發者,見識了很多一流大公司對之的態度,在很多應用方面幾乎看不到影子,深深感覺發展前景不是很樂觀,但現在不同以往跨平臺,加 ...
  • IAuthenticationFilter是MVC5中的新特性,它有2個關鍵方法: OnAuthentication OnAuthenticationChallenge 當IAuthenticationFilter和IAuthorizationFilter結合使用時,流程看似比較複雜: 在OnAut ...
  • 官方的文檔https://docs.asp.net/en/latest/tutorials/first-mvc-app/start-mvc.html 先來看一下實現的效果 開始之前,確定本機已經有.NET Core環境。https://www.microsoft.com/net/core#windo ...
  • C 可以直接引用C++的DLL和轉換JAVA寫好的程式。最近由於工作原因接觸這方面比較多,根據實際需求,我們通過一個具體例子把一個JAVA方法轉換成可以由C 直接調用的DLL C 調用c++ C 調用C++的例子網上很多,以一個C++的具體方法為例。 C++代碼 C 代碼 這樣我們把這個DLL放在程 ...
  • 一、struts.xml中<package>的namespace屬性的用法 在實際的開發中常會遇到name相同的<action>,如下代碼: 以上的配置中在同一個namespace下有兩個相同name為add的<action>,這樣就不能區別了,為瞭解決這個問題,我們可以把兩個<action>放在不 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...