青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

ACM___________________________

______________白白の屋
posts - 182, comments - 102, trackbacks - 0, articles - 0
<2011年4月>
272829303112
3456789
10111213141516
17181920212223
24252627282930
1234567

常用鏈接

留言簿(24)

隨筆分類(332)

隨筆檔案(182)

FRIENDS

搜索

積分與排名

最新隨筆

最新評論

閱讀排行榜

評論排行榜

并查集 學(xué)習(xí) 詳解

Posted on 2010-08-10 11:02 MiYu 閱讀(2868) 評論(2)  編輯 收藏 引用 所屬分類: ACM ( 并查集 )

 

并查集--學(xué)習(xí)詳解

文章作者:yx_th000 文章來源:Cherish_yimi (http://www.cnblogs.com/cherish_yimi/)

l         并查集:(union-find sets)

一種簡單的用途廣泛的集合. 并查集是若干個不相交集合,能夠?qū)崿F(xiàn)較快的合并和判斷元素所在集合的操作,應(yīng)用很多,如其求無向圖的連通分量個數(shù)等。最完美的應(yīng)用當(dāng)屬:實現(xiàn)Kruskar算法求最小生成樹。

l         并查集的精髓(即它的三種操作,結(jié)合實現(xiàn)代碼模板進(jìn)行理解):

1、Make_Set(x) 把每一個元素初始化為一個集合

初始化后每一個元素的父親節(jié)點(diǎn)是它本身,每一個元素的祖先節(jié)點(diǎn)也是它本身(也可以根據(jù)情況而變)。

2、Find_Set(x) 查找一個元素所在的集合

查找一個元素所在的集合,其精髓是找到這個元素所在集合的祖先!這個才是并查集判斷和合并的最終依據(jù)。
判斷兩個元素是否屬于同一集合,只要看他們所在集合的祖先是否相同即可。
合并兩個集合,也是使一個集合的祖先成為另一個集合的祖先,具體見示意圖

3Union(x,y) 合并x,y所在的兩個集合

合并兩個不相交集合操作很簡單:
利用Find_Set找到其中兩個集合的祖先,將一個集合的祖先指向另一個集合的祖先。如圖

l         并查集的優(yōu)化

1、Find_Set(x)時 路徑壓縮
尋找祖先時我們一般采用遞歸查找,但是當(dāng)元素很多亦或是整棵樹變?yōu)橐粭l鏈時,每次Find_Set(x)都是O(n)的復(fù)雜度,有沒有辦法減小這個復(fù)雜度呢?
答案是肯定的,這就是
路徑壓縮,即當(dāng)我們經(jīng)過"遞推"找到祖先節(jié)點(diǎn)后,"回溯"的時候順便將它的子孫節(jié)點(diǎn)都直接指向祖先,這樣以后再次Find_Set(x)時復(fù)雜度就變成O(1)了,如下圖所示;可見,路徑壓縮方便了以后的查找。

2、Union(x,y)時 按秩合并
即合并的時候?qū)⒃厣俚募虾喜⒌皆囟嗟募现校@樣合并之后樹的高度會相對較小。

l         主要代碼實現(xiàn)

Code
 1 int father[MAX];   /**//* father[x]表示x的父節(jié)點(diǎn)*/
 2 int rank[MAX];     /**//* rank[x]表示x的秩*/
 3
 4
 5 /**//* 初始化集合*/
 6 void Make_Set(int x)
 7 {
 8     father[x] = x; //根據(jù)實際情況指定的父節(jié)點(diǎn)可變化
 9     rank[x] = 0;   //根據(jù)實際情況初始化秩也有所變化
10 }
11
12
13 /**//* 查找x元素所在的集合,回溯時壓縮路徑*/
14 int Find_Set(int x)
15 {
16     if (x != father[x])
17      {
18         father[x] = Find_Set(father[x]); //這個回溯時的壓縮路徑是精華
19     }
20     return father[x];
21 }
22
23
24 /**//* 
25    按秩合并x,y所在的集合
26    下面的那個if else結(jié)構(gòu)不是絕對的,具體根據(jù)情況變化
27    但是,宗旨是不變的即,按秩合并,實時更新秩。
28 */
29 void Union(int x, int y)
30 {
31     x = Find_Set(x);
32     y = Find_Set(y);
33     if (x == y) return;
34     if (rank[x] > rank[y]) 
35      {
36         father[y] = x;
37     }
38     else
39      {
40         if (rank[x] == rank[y])
41          {
42             rank[y]++;
43         }
44         father[x] = y;
45     }
46 }
47

 

注:學(xué)習(xí)并查集時非常感謝Slyar提供的資料,這里注明鏈接:http://www.slyar.com/blog/
另外,我認(rèn)為寫并查集時涉及到的路徑壓縮,最好用遞歸,一方面代碼的可讀性非常好,另一方面,可以更直觀的理解路徑壓縮時在回溯時完成的巧妙。

 

 

PKU POJ 1161解題報告(并查集)

 

Time Limit: 1000MS

 

Memory Limit: 20000K

Total Submissions: 5572

 

Accepted: 2660

Description

Severe acute respiratory syndrome (SARS), an atypical pneumonia of unknown aetiology, was recognized as a global threat in mid-March 2003. To minimize transmission to others, the best strategy is to separate the suspects from others.
In the Not-Spreading-Your-Sickness University (NSYSU), there are many student groups. Students in the same group intercommunicate with each other frequently, and a student may join several groups. To prevent the possible transmissions of SARS, the NSYSU collects the member lists of all student groups, and makes the following rule in their standard operation procedure (SOP).
Once a member in a group is a suspect, all members in the group are suspects.
However, they find that it is not easy to identify all the suspects when a student is recognized as a suspect. Your job is to write a program which finds all the suspects.

Input

The input file contains several cases. Each test case begins with two integers n and m in a line, where n is the number of students, and m is the number of groups. You may assume that 0 < n <= 30000 and 0 <= m <= 500. Every student is numbered by a unique integer between 0 and n−1, and initially student 0 is recognized as a suspect in all the cases. This line is followed by m member lists of the groups, one line per group. Each line begins with an integer k by itself representing the number of members in the group. Following the number of members, there are k integers representing the students in this group. All the integers in a line are separated by at least one space.
A case with n = 0 and m = 0 indicates the end of the input, and need not be processed.

Output

For each case, output the number of suspects in one line.

Sample Input

100 4

2 1 2

5 10 13 11 12 14

2 0 1

2 99 2

200 2

1 5

5 1 2 3 4 5

1 0

0 0

Sample Output

4

1

1

我的思路:

典型的并查集,最初各自為集,然后每個group進(jìn)行合并,等到所有的group合并完,題目也就解決了,因為在合并的時候,如果哪兩個group中有重合的元素,則那個后來的group會由于按秩合并的原則自動合并到

先有的集合當(dāng)中,奧妙便在其中。下面是代碼:

Code
 1 #include<iostream>
 2 using namespace std;
 3
 4 int n, m, i, j;
 5 int father[30005], num[30005];
 6
 7 void makeSet(int n)
 8 {
 9     for(i = 0; i < n; i++)
10      {
11         father[i] = i; //使用本身做根
12         num[i] = 1;
13     }
14 }
15 int findSet(int x)
16 {
17     if(father[x] != x) //合并后的樹的根是不變的
18      {    
19         father[x] = findSet(father[x]);
20     }
21     return father[x]; 
22 }
23
24 void Union(int a, int b)
25 {
26     int x = findSet(a);
27     int y = findSet(b);
28     if(x == y)
29      {
30         return;
31     }
32     if(num[x] <= num[y])
33      {
34         father[x] = y;
35         num[y] += num[x];
36     }
37     else 
38      {
39         father[y] = x;
40         num[x] += num[y];
41     }
42 }
43
44 int main()
45 {
46     while(scanf("%d %d", &n, &m)!=EOF && n != 0)
47      {
48         makeSet(n);
49         for(i = 0; i < m; i++)
50          {
51             int count, first, b;
52             scanf("%d %d",&count, &first);
53             for(j = 1; j < count; j++)
54              {
55                 scanf("%d",&b);
56                 Union(first,b);
57             }
58         }
59         printf("%d\n",num[findSet(0)]);
60     }
61     return 0;
62 }
63

另外,上面并查集的根我是采用數(shù)字本身的,然后路徑壓縮尋找父親節(jié)點(diǎn)是采用遞歸的,下面貼出,采用-1做根和使用非遞歸實現(xiàn)的部分代碼:

Code
 1 void makeSet(int n)
 2 {
 3     for(i = 0; i < n; i++)
 4      {
 5         father[i] = -1;
 6         num[i] = 1;
 7     }
 8 }
 9 //非遞歸實現(xiàn)
10 int findSet(int x)
11 {
12     while(father[x] != -1)   //根為-1
13      {
14         x = father[x];
15     }
16     return x;
17 }
18

 

PKU POJ 2524 解題報告(并查集)


 

Ubiquitous Religions

Time Limit: 5000MS

 

Memory Limit: 65536K

Total Submissions: 9637

 

Accepted: 4463

Description

There are so many different religions in the world today that it is difficult to keep track of them all. You are interested in finding out how many different religions students in your university believe in.

You know that there are n students in your university (0 < n <= 50000). It is infeasible for you to ask every student their religious beliefs. Furthermore, many students are not comfortable expressing their beliefs. One way to avoid these problems is to ask m (0 <= m <= n(n-1)/2) pairs of students and ask them whether they believe in the same religion (e.g. they may know if they both attend the same church). From this data, you may not know what each person believes in, but you can get an idea of the upper bound of how many different religions can be possibly represented on campus. You may assume that each student subscribes to at most one religion.


Input

The input consists of a number of cases. Each case starts with a line specifying the integers n and m. The next m lines each consists of two integers i and j, specifying that students i and j believe in the same religion. The students are numbered 1 to n. The end of input is specified by a line in which n = m = 0.


Output

For each test case, print on a single line the case number (starting with 1) followed by the maximum number of different religions that the students in the university believe in.


Sample Input
10 9
1 2
1 3
1 4
1 5
1 6
1 7
1 8
1 9
1 10
10 4
2 3
4 5
4 8
5 8
0 0

Sample Output

Case 1: 1

Case 2: 7

我的思路:

有了前面學(xué)習(xí)的基礎(chǔ)和1161的練習(xí),這道題完全就可以水過了,設(shè)定一個計數(shù)器,每次合并時減1便可值得一提的是:樹的秩是好東西啊,既可以記錄每個集合的節(jié)點(diǎn)個數(shù),又能保證按秩合并成相對高度小的樹。

代碼如下:

Code
 1 #include<iostream>
 2 using namespace std;
 3
 4 int n, m, maxNum, i;
 5 int father[50005], num[50005];
 6
 7 void makeSet(int n)
 8 {
 9     for(int j = 1; j <= n; j++)
10      {
11         father[j] = j;
12         num[j] = 1;
13     }
14 }
15 int findSet(int x)
16 {
17     if(father[x] != x) 
18      {    
19         father[x] = findSet(father[x]);
20     }
21     return father[x]; 
22 }
23
24 void Union(int a, int b)
25 {
26     int x = findSet(a);
27     int y = findSet(b);
28     if(x == y)
29      {
30         return;
31     }
32     if(num[x] <= num[y])
33      {
34         father[x] = y;
35         num[y] += num[x];
36         maxNum--;
37     }
38     else 
39      {
40         father[y] = x;
41         num[x] += num[y];
42         maxNum--;
43     }
44 }
45
46 int main()
47 {
48     int Case = 1;
49     while(scanf("%d %d", &n, &m)!=EOF && n!=0)
50      {
51         maxNum = n;
52         makeSet(n);
53         int a, b;
54         for(i = 0; i < m; i++)
55          {
56             scanf("%d %d",&a, &b);
57             Union(a,b);
58         }
59         printf("Case %d: %d\n",Case++,maxNum);
60     }
61     return 0;
62 }
63

POJ 1611 The Suspects          最基礎(chǔ)的并查集

 

POJ 2524 Ubiquitous Religions 最基本的并查集

POJ 1182 食物鏈       并查集的拓展

注意: 只有一組數(shù)據(jù);

要充分利用題意所給條件:有三類動物A,B,C,這三類動物的食物鏈

構(gòu)成了有趣的環(huán)形。AB BCCA。也就是說:只有三個group

POJ 2492 A Bug's Life 并查集的拓展

法一:深度優(yōu)先遍歷

每次遍歷記錄下該點(diǎn)是男還是女,只有:-〉女,女-〉男滿足,否則,找到同性戀,結(jié)束程序。

法二:二分圖匹配

法三:并查集的拓展:1182很像,只不過這里就有兩組,而1182是三組,1611無限制

POJ 1861 Network == zju_1542    并查集+自定義排序+貪心求"最小生成樹"

答案不唯一,不過在ZOJ上用QSORT()SORT()都能過,在POJ上只有SORT()才能過...

POJ 1703 Find them, Catch them 并查集的拓展

這個和POJ 2492 A Bug's Life很像,就是把代碼稍微修改了一下就AC了!

注意:And of course, at least one of them belongs to Gang Dragon, and the same for Gang Snake. 就是說只有兩個組。

POJ 2236 Wireless Network        并查集的應(yīng)用

需要注意的地方:1、并查集;2、N的范圍,可以等于10013、從N+1行開始,第一個輸入的可以是字符串。

POJ 1988 Cube Stacking            并查集很好的應(yīng)用

1、與 銀河英雄傳說==NOI2002 Galaxy一樣;2、增加了一個數(shù)組behind[x],記錄戰(zhàn)艦x在列中的相對位置;3、詳細(xì)解題報告見銀河英雄傳說。

 

JOJ 1905 Freckles   == POJ 2560 最小生成樹

 

法一:Prim算法;法二:并查集實現(xiàn)Kruskar算法求最小生成樹

Feedback

# re: 并查集 學(xué)習(xí) 詳解  回復(fù)  更多評論   

2011-03-03 19:48 by 貝殼里的海
學(xué)習(xí)了,內(nèi)容非常豐富~~~~

# re: 并查集 學(xué)習(xí) 詳解  回復(fù)  更多評論   

2011-04-30 16:35 by 游客
非常有用 感謝分享
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            国模私拍一区二区三区| 午夜精品久久久久久久99黑人| 欧美日韩视频在线| 亚洲欧美日韩在线综合| 亚洲欧美综合国产精品一区| 国语精品一区| 亚洲国产成人av好男人在线观看| 欧美欧美天天天天操| 亚洲欧美韩国| 久久精品一区四区| 亚洲国产一区二区三区a毛片| 亚洲精品视频在线| 国产女人水真多18毛片18精品视频| 久久久久国产一区二区| 欧美大学生性色视频| 亚洲免费一区二区| 久久久久久久久岛国免费| 日韩一级成人av| 亚洲免费一在线| 亚洲国产日韩欧美在线99 | 久久精品二区三区| 最新日韩av| 亚洲一区二区三区四区中文| 在线观看91精品国产麻豆| 亚洲精品黄色| 国产亚洲精品福利| 亚洲激情啪啪| 国产日本亚洲高清| 亚洲国产美国国产综合一区二区| 国产精品推荐精品| 欧美激情亚洲自拍| 国产精品自在欧美一区| 亚洲高清二区| 国产深夜精品福利| 亚洲日本中文字幕区| 国产在线乱码一区二区三区| 亚洲精品国产无天堂网2021| 国内精品久久久久影院色| 亚洲裸体在线观看| 禁久久精品乱码| 一区二区三区鲁丝不卡| 亚洲高清色综合| 亚洲欧美春色| 一区二区av在线| 久久婷婷影院| 欧美一级日韩一级| 欧美精品乱码久久久久久按摩| 久久久久久午夜| 国产精品99一区二区| 欧美国产精品一区| 国产热re99久久6国产精品| 亚洲三级色网| 久久综合给合久久狠狠色| 欧美日韩成人激情| 久久亚洲电影| 国产精品卡一卡二卡三| 亚洲人成网站影音先锋播放| 欧美激情一区二区三级高清视频| 国产精品视频999| 亚洲精品国偷自产在线99热| 一区二区三区自拍| 亚洲欧美精品在线观看| 一区二区三区四区五区在线| 久久综合999| 久久精品亚洲一区二区三区浴池| 欧美午夜理伦三级在线观看| 亚洲国产欧美另类丝袜| 在线观看中文字幕亚洲| 欧美亚洲三区| 午夜欧美大片免费观看| 欧美日韩不卡在线| 亚洲国产美女| 亚洲国产精品传媒在线观看| 久久精品30| 久久久www成人免费精品| 国产精品亚发布| 这里只有精品丝袜| 国产精品99久久99久久久二8| 免费在线成人av| 麻豆精品在线视频| 狠狠狠色丁香婷婷综合久久五月| 午夜精品久久久久久久男人的天堂| 亚洲永久网站| 欧美色播在线播放| 亚洲精品视频在线播放| 日韩一级黄色av| 欧美a级一区二区| 欧美激情bt| 亚洲国产日韩欧美一区二区三区| 久久久亚洲精品一区二区三区 | 国产精品入口福利| 一区二区三区视频观看| 中文欧美日韩| 欧美日在线观看| 日韩视频―中文字幕| 99精品国产热久久91蜜凸| 欧美大尺度在线| 亚洲国产一区二区a毛片| 亚洲三级免费电影| 欧美精品一区二区蜜臀亚洲| 亚洲人成网站在线播| 9l国产精品久久久久麻豆| 欧美精品一区二区三区在线看午夜| 亚洲欧洲精品一区| 一本色道久久88综合亚洲精品ⅰ| 欧美日韩成人综合在线一区二区 | 亚洲美女毛片| 欧美精品免费观看二区| 日韩视频免费| 亚洲综合视频在线| 国产精品一区二区女厕厕| 午夜一区二区三区在线观看| 久久久久久久波多野高潮日日 | 亚洲毛片视频| 亚洲一区综合| 国产欧美一区二区精品性色| 欧美综合国产| 欧美高清在线精品一区| 亚洲美女av网站| 欧美视频免费在线| 亚洲在线第一页| 久久人体大胆视频| 亚洲国产日韩一级| 欧美理论电影在线播放| 亚洲视频1区2区| 久久久久久亚洲精品中文字幕| 影音先锋在线一区| 欧美激情中文字幕一区二区| 一本色道久久99精品综合 | 亚洲性人人天天夜夜摸| 国产精品推荐精品| 久久久www成人免费毛片麻豆| 欧美成人综合网站| 亚洲视频精品在线| 国产日韩欧美一区二区三区四区| 久久久久久久久伊人| 亚洲黄色av一区| 亚洲欧美清纯在线制服| 国内精品久久久久久久影视蜜臀 | 欧美一区二区三区免费在线看| 国产亚洲视频在线| 你懂的国产精品| 中日韩男男gay无套| 久久久久亚洲综合| 99re66热这里只有精品3直播 | 欧美日韩中字| 欧美一区二区播放| 亚洲国产精品高清久久久| 亚洲综合精品四区| 一区二区三区在线视频播放| 欧美—级高清免费播放| 亚洲欧美日本日韩| 亚洲电影免费观看高清| 亚洲欧美日韩综合一区| 136国产福利精品导航网址应用| 欧美日韩的一区二区| 小黄鸭精品密入口导航| 亚洲激情在线观看视频免费| 欧美在线日韩| 日韩视频免费| 国产在线播放一区二区三区| 欧美极品一区| 欧美一区影院| 亚洲精品一区二区三区不| 久久久99精品免费观看不卡| av成人老司机| 国内偷自视频区视频综合| 欧美日韩在线直播| 久久亚洲精品欧美| 亚洲无吗在线| 亚洲国产精品成人一区二区| 久久精品国产欧美亚洲人人爽| aa国产精品| 在线欧美不卡| 国产精品视频yy9299一区| 欧美成人福利视频| 久久xxxx| 亚洲天堂免费观看| 亚洲激情六月丁香| 免费h精品视频在线播放| 亚洲欧美激情视频| 亚洲狼人综合| 伊人久久亚洲热| 国产精品亚洲综合一区在线观看| 欧美精品aa| 久久全国免费视频| 亚洲欧美在线视频观看| 99视频超级精品| 亚洲国产1区| 麻豆freexxxx性91精品| 欧美一区二区三区成人| 中文在线资源观看网站视频免费不卡 | 久久激情视频久久| 亚洲一区在线播放| 亚洲美女精品成人在线视频| 在线观看视频一区| 国产一区二区三区久久| 国产精品日本一区二区| 欧美日韩亚洲免费| 欧美黑人国产人伦爽爽爽|