Problem : https://www.acmicpc.net/problem/16724
First Approach
- 처음 문제를 보고 단순 BFS 인 줄 알았다. 하지만 시작점에 따라 집합의 개수가 달라진다.
-
// ex 2 4 DLDL RLRL // 단순 BFS로 (0,0)부터 시작하면 4를 뱉는다. 정답은 2이다.
Approach
집합(사이클)의 개수가 몇 개인지를 찾는 문제이다. Union-Find 알고리즘을 사용하였다.
범위를 벗어나는 입력이 없으므로, 무조건 1집합 당 1사이클이 생긴다.
- 현재 위치의 값 (U, D, L, R) 을 보고, 다음 위치를 구한다.
- 두 위치를 union 한다.
마지막으로 집합의 개수를 센다.(나는 set 에 최종 부모를 넣고 set의 사이즈를 출력했다.)
Code
import java.io.*;
import java.util.HashSet;
import java.util.StringTokenizer;
/**
* No.16725: 피리 부는 사나이
* URL: https://www.acmicpc.net/problem/16724
* Hint: 집합(사이클)이 몇개인지를 판단하는 문제 -> 유니온파인드 사용
*/
public class BOJ16724 {
static int n, m;
static char[][] map;
static int[] parents;
public static void main(String[] args) throws IOException {
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(System.out));
StringTokenizer st = new StringTokenizer(br.readLine());
n = Integer.parseInt(st.nextToken());
m = Integer.parseInt(st.nextToken());
map = new char[n][m];
parents = new int[n * m];
for (int i = 0; i < parents.length; i++) {
parents[i] = i;
}
for (int i = 0; i < n; i++) {
map[i] = br.readLine().toCharArray();
}
for (int i = 0; i < n; i++) {
for (int j = 0; j < m; j++) {
int curIdx = i * m + j;
int nextIdx = findIdx(i, j);
if (find(curIdx) != find(nextIdx)) {
union(curIdx, nextIdx);
}
}
}
int ans = findSaveZone();
bw.write(String.valueOf(ans));
bw.close();
br.close();
}
// 집합의 개수를 카운트
static int findSaveZone() {
HashSet<Integer> set = new HashSet<>();
for (int i = 0; i < parents.length; i++) {
set.add(find(i));
}
return set.size();
}
// parents 배열이 1차원이기 때문에 2차원배열을 1차원배열로 인덱싱
static int findIdx(int x, int y) {
int nx = x;
int ny = y;
if (map[x][y] == 'U') {
nx--;
} else if (map[x][y] == 'D') {
nx++;
} else if (map[x][y] == 'L') {
ny--;
} else if (map[x][y] == 'R') {
ny++;
}
return nx * m + ny;
}
// 주어진 파라미터의 최상위 부모를 찾음
static int find(int idx) {
if (parents[idx] == idx) {
return idx;
}
return parents[idx] = find(parents[idx]);
}
// 주어진 파라미터들을 묶음
static void union(int idx1, int idx2) {
idx1 = find(idx1);
idx2 = find(idx2);
if (idx1 > idx2) {
parents[idx1] = idx2;
} else {
parents[idx2] = idx1;
}
}
}
'Algorithm > Baekjoon Online Judge' 카테고리의 다른 글
[Java] 백준 16946 (벽 부수고 이동하기 4) Gold 2 (0) | 2021.04.09 |
---|---|
[Java] 백준 17143 (낚시왕) Gold 2 (0) | 2021.04.09 |
[Java] 백준 13460 (구슬 탈출 2) Gold 2 (0) | 2021.04.07 |
[Java] 백준 1766 (문제집) Gold 2 (0) | 2021.04.07 |
[Java] 백준 12100 (2048 Easy) Gold 2 (0) | 2021.04.05 |