PS/백준

[BOJ] 백준 10216번 : Count Circle Groups (JAVA)

제이온 (Jayon) 2020. 8. 5.

문제

백준이는 국방의 의무를 수행하기 위해 떠났다. 혹독한 훈련을 무사히 마치고 나서, 정말 잘 생겼고 코딩도 잘하는 백준은 그 특기를 살려 적군의 진영을 수학적으로 분석하는 일을 맡게 되었다.

 

2차원 평면 위의 N곳에 적군의 진영이 설치되어 있다. 각 적군의 진영들은 진영마다 하나의 통신탑을 설치해, i번째 적군의 통신탑은 설치 위치로부터 Ri 이내 거리에 포함되는 모든 지역을 자신의 통신영역 Ai로 가지게 된다. 

 

만약 임의의 통신영역 Ai와 Aj가 닿거나 겹치는 부분이 있다면 진영 i와 진영 j는 직접적으로 통신이 가능하다. 물론 직접적으로 통신이 가능하지 않더라도, 임의의 지역 i와 j가 중간에 몇 개의 직접통신을 거쳐서 최종적으로 통신이 가능하다면 i와 j는 상호간에 통신이 가능한 것으로 본다.

 

적들은 영리해서, 상호간에 통신이 가능한 부대끼리는 결집력있는 한 그룹처럼 행동한다. 백준은 이러한 그룹의 개수를 알아내 아군의 전략지침에 도움을 주고자 한다. 군대에 가서도 코딩하는 불쌍한 백준을 위해 적군의 통신망 분석을 도와주자!

 

 

입력

입력 파일의 첫 번째 줄에 테스트 케이스의 수를 의미하는 자연수 T가 주어진다. 그 다음에는 T개의 테스트 케이스가 주어진다.

 

각각의 테스트 케이스에 대해서 적군 진영의 숫자 N (1 ≤ N ≤ 3,000)이 주어진다. 이어서 N줄에 걸쳐 적군 진영의 좌표 x, y (0 ≤ x, y ≤ 5,000), 그리고 해당 진영의 R (0 ≤ R ≤ 5,000)이 주어진다. 주어지는 수는 모두 정수이다.

 

 

출력

각 테스트 케이스에 대해서 한 줄에 걸쳐 적군 진영의 그룹 개수를 출력한다.

 

 

풀이

유니온 파인드를 이용하여 풀 수 있는 문제였습니다. 전반적인 로직은 아래와 같습니다.

 

 

1. 유니온 파인드용 루트 노드 parent 1차원 배열을 정의하고, (x, y) 좌표와 r 거리를 담는 unit 2차원 배열을 정의한다.

 

2. 처음에는 모두 독자적인 1인 그룹을 형성하므로 ans = N으로 초기화한다.

 

3. 만약, 특정 두 정점 사이의 거리가 두 정점의 r을 합한 값보다 작거나 같다면, 통신 범위가 겹친다는 의미이므로 두 정점을 합집합 연산을 한다. 그리고 1인 그룹 2개가 하나의 그룹으로 재결합한 것이기때문에 ans에서 1을 뺀다.

 

4. 3번 과정을 반복한다.

 

 

아래는 위 과정을 정리한 소스코드입니다.

 

 

소스코드

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.util.StringTokenizer;
 
public class Main {
    static int[] parent; // 유니온 파인드 루트 노드 배열
    static int[][] unit; // unit[x][y]의 거리는 r이다.
 
    public static void main(String[] args) throws NumberFormatException, IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(System.out));
        StringTokenizer st;
 
        int T = Integer.parseInt(br.readLine());
 
        StringBuilder sb = new StringBuilder();
        while (T-- > 0) {
            int N = Integer.parseInt(br.readLine());
 
            parent = new int[N];
            unit = new int[N][3];
 
            for (int i = 0; i < N; i++) {
                st = new StringTokenizer(br.readLine());
 
                int x = Integer.parseInt(st.nextToken());
                int y = Integer.parseInt(st.nextToken());
                int r = Integer.parseInt(st.nextToken());
 
                unit[i][0= x;
                unit[i][1= y;
                unit[i][2= r;
 
                parent[i] = i;
            }
 
            int ans = N; // 처음에는 모두 1인 그룹을 형성.
            for (int i = 0; i < N; i++) {
                for (int j = i + 1; j < N; j++) {
                    int x_dif = unit[i][0- unit[j][0];
                    int y_dif = unit[i][1- unit[j][1];
                    int r = unit[i][2+ unit[j][2];
 
                    // 특정 두 정점의 통신 범위가 겹치는 경우
                    if (x_dif * x_dif + y_dif * y_dif <= r * r) {
                        if (find(i) != find(j)) {
                            union(i, j); // 합집합 연산 수행
                            ans--// 1인 그룹 2개가 하나의 그룹을 형성하므로 ans를 1 줄여야 함.
                        }
                    }
                }
            }
 
            sb.append(ans + "\n");
        }
 
        bw.write(sb.toString());
        bw.flush();
        bw.close();
        br.close();
    }
 
    public static int find(int x) {
        if (x == parent[x]) {
            return x;
        }
 
        return parent[x] = find(parent[x]);
    }
 
    public static void union(int x, int y) {
        x = find(x);
        y = find(y);
 
        if (x != y) {
            if (x < y) {
                parent[y] = x;
            } else {
                parent[x] = y;
            }
        }
    }
 
}
cs

 

지적 혹은 조언 환영합니다! 언제든지 댓글로 남겨주세요.

댓글0

추천 글