hdu 3172 Virtual Friends(并查集)

题目比较简单,但作为长久不写题之后的热身题还是不错的。

统计每组朋友的朋友圈的大小。

如果a和b是朋友,这个朋友圈的大小为2,如果b和c也是朋友,那么a和c也是朋友,此时这个朋友圈的大小为3。

输入t,表示接下来有t组数据。

每组数据有n组朋友关系。

接下来n行,每行一组朋友关系,然后输出这组朋友的朋友圈大小,即有多少朋友。

然后又是t组数据……(这点好坑)重复上述输入,直到数据结束。

因为最多有10^5个人,那么如果用线性字符串数组保存人名,肯定超时得不要不要的,所以要用map(每次操作时间复杂度为log2(n))。

据说如果并查集不压缩也会爆。我没试,诸位可以试一下。

废话说完,上代码

 1 #include <cstdio>
 2 #include <cstring>
 3 #include <cmath>
 4 #include <algorithm>
 5 #include <map>
 6 #include <string>
 7 using namespace std;
 8
 9 map<string, int> mp;                //字符串与数字转换
10 int val[100010];                    //朋友数
11 int mg[100010];                     //并查集使用的数组
12 int n, t;
13
14 int find(int x)
15 {
16     int fx = x;
17     while(mg[fx] != fx) fx = mg[fx];
18     while(mg[x] != x)
19     {
20         int mid = mg[x];
21         mg[x] = fx;
22         x = mid;
23     }
24     //printf("%5d\n", fx);
25     return fx;
26 }
27
28 void merge(int x, int y)
29 {
30     int mx = find(x);
31     int my = find(y);
32     if(mx != my)
33     {
34         val[mx] += val[my];
35         val[my] = val[mx];
36         mg[mx] = my;
37
38     }
39     printf("%d\n", val[mx]);
40 }
41
42 int main()
43 {
44 //    freopen("test.txt", "r", stdin);
45     while(~scanf("%d", &t))
46     {
47         while(t--)
48         {
49             scanf("%d", &n);
50             for(int i = 0; i < 100010; i++)
51             {
52                 val[i] = 1;
53                 mg[i] = i;
54             }
55             char a[25], b[25];
56             int ans = 1;                            //因为map中int初始值为0,所以赋值时应从1开始
57             mp.clear();                             //每次注意清空mp
58             for(int i = 0; i < n; i++)
59             {
60                 scanf("%s%s", a, b);
61                 if(!mp[a]) mp[a] = ans++;           //加入新节点
62                 if(!mp[b]) mp[b] = ans++;
63                 merge(mp[a], mp[b]);
64             }
65         }
66     }
67     return 0;
68 }
时间: 2024-10-09 21:29:44

hdu 3172 Virtual Friends(并查集)的相关文章

hdu 3172 Virtual Friends (并查集 + 字典树)

题目: 链接:点击打开链接 题意: 输入n,给出n行数据,每行有两个字符串,输出关系网络中朋友的个数,n行. 思路: 代码: #include <iostream> #include <cstdio> #include <cstdlib> #include <cstring> using namespace std; const int N = 22; const int M = 200020; struct node { int c; node *chil

HDU 3172 Virtual Friends(带权并查集)

题目地址:HDU 3172 带权并查集水题.每次合并的时候维护一下权值.注意坑爹的输入.. 代码如下: #include <iostream> #include <cstdio> #include <string> #include <cstring> #include <stdlib.h> #include <math.h> #include <ctype.h> #include <queue> #inclu

HDU 3172 Virtual Friends 带权并查集 -秩

ll T; while(~scanf("%d",&T)){ while(T--) { = = ... 思路: 用秩合并,看了题解才发现 if(fx == fy)要输出当前集合的秩而不是0... #include <cstdio> #include <iostream> #include <algorithm> #include <string.h> #include <vector> #include <map&

hdu 3234 Exclusive-OR (并查集+异或性质)

Exclusive-OR Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 2177    Accepted Submission(s): 603 Problem Description You are not given n non-negative integers X0, X1, ..., Xn-1 less than 220 ,

HDU 4496 D-City (并查集)

题意:给你n个点m条边,问删除前i条边后有多少个连通分块. 思路:从后往前操作,从后往前添加i条边等于添加完m条边后删掉前m-i条边,可知刚开始没有边,所以sum[m]=n; #include <stdio.h> #include <iostream> #include <algorithm> #include <string.h> #include <queue> #include <math.h> #define M 100010

hdu 1811Rank of Tetris (并查集 + 拓扑排序)

1 /* 2 题意:这些信息可能有三种情况,分别是"A > B","A = B","A < B",分别表示A的Rating高于B,等于B,小于B. 3 4 现在Lele并不是让你来帮他制作这个高手榜,他只是想知道,根据这些信息是否能够确定出这个高手榜,是的话就输出"OK". 5 否则就请你判断出错的原因,到底是因为信息不完全(输出"UNCERTAIN"),还是因为这些信息中包含冲突(输出&quo

HDU 1232 畅通工程(并查集)

畅通工程 Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submission(s): 30485    Accepted Submission(s): 16013 Problem Description 某省调查城镇交通状况,得到现有城镇道路统计表,表中列出了每条道路直接连通的城镇.省政府"畅通工程"的目标是使全省任何两个城镇间都可以实现交通(但不一定有

HDU 2017 Code Lock (并查集的应用+快速幂)

链接:HDU 3461 题目大意: 题目的大意是一个密码锁上有编号为1到N的N个字母,每个字母可以取26个小写英文字母中的一个.再给你M个区间[L,M],表示该区间的字母可以一起同步"增加"(从'a'变为'b'为增1,'z'增1为'a').假如一组密码按照给定的区间进行有限次的"增加"操作后可以变成另一组密码,那么我们认为这两组密码是相同的.该题的目标就是在给定N.M和M个区间的前提下计算有多少种不同的密码. 根据题意,如果一个可调整的区间都没有的话,答案应该是26

HDU 4775 Infinite Go(并查集,模拟)

HDU 4775 Infinite Go 题目链接 题意:围棋,两人轮流走,如果有一链被围死,就会被吃掉,问下完后最后黑色和白色各剩多少棋 思路:模拟,利用一个并查集来保存链,然后并记录下周围有多少个空格,然后去模拟,注意几个点,就是删除的时候,要把空格还回去,还有边界的位置是也算被围死的 代码: #include <stdio.h> #include <string.h> #include <queue> #include <map> using name

HDU 1198-Farm Irrigation(并查集)

Farm Irrigation Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submission(s): 5820    Accepted Submission(s): 2523 Problem Description Benny has a spacious farm land to irrigate. The farm land is a rectangle