-
Q. 2660 자바 : 회장 뽑기코딩테스트/백준_Java 2023. 4. 20. 16:50
Q. 2660 자바 : 회장 뽑기(BFS)
문제 : boj.kr/2660
골드 5 난이도의 문제이다.
간선들이 주어지고, 모든 정점에 방문 가능한 깊이가 가장 적은 사람을 구하는 문제이다.
주의해야 할 점은 출력할 때 회장 후보를 오름차순으로 모두 출력한다는 내용 때문이 있으므로
결과값을 계산한 뒤 정렬이 한번 필요하다.
아래와 같은 방식으로 풀었다.
1. 모든 정점을 계산해야 하므로 모든 정점에 대해 bfs 탐색이 필요하다.
2. 탐색을 할 때 만약 깊이를 줄일 수 있다면 방문한 정점도 다시 탐색한다.
(이전 정점을 탐색하지 않도록 주의)
3. 탐색이 끝난 뒤 모든 정점 방문 시 깊이에 대한 값을 계산한다.
이 때 회장 후보일 수 있으므로 결과값과 현재 회장 후보의 값을 비교해 저장한다.
4. 깊이의 최소값과 일치하는 정점을 구하고, 정렬해서 출력한다.
시간 복잡도의 경우 아래와 같다.
1. 모든 정점에 대한 bfs 탐색 - O(N * N^2 = N^3)
2. 각 정점마다 bfs 탐색 뒤 깊이 계산 - O(N)
3. 모든 정점 탐색 뒤 최소값 계산 - O(N)
4. 최소값과 일치하는 정점 계산 뒤 정렬 - O(N log N)
5. 출력 - O(N)
복잡해 보이지만, 모든 시간 복잡도는 연관이 없으므로 더해서 계산하면 된다.
그래서 시간복잡도는 약 O(N^3)이 나와 시간복잡도는 긴 편 이지만,
N의 최대값이 50이므로 시간초과는 걱정하지 않아도 된다.
풀 때 골드5문제 치고 꽤나 구현량이 많아서 찾아 보니
플로이드-와샬 알고리즘을 이용해서 푸는 문제라고 한다.
그리고 이 알고리즘을 사용해도 시간 복잡도가 O(N^3)이다.
풀이는 아래와 같다.
import java.io.*; import java.util.*; public class Main { static FastReader scan = new FastReader(); static StringBuilder sb = new StringBuilder(); static int n, min; static ArrayList<Integer>[] Graph; static int[] Cnt; static void input(){ n = scan.nextInt(); min = n; Graph = new ArrayList[n + 1]; Cnt = new int[n + 1]; for (int i = 1; i <= n; i++) { Graph[i] = new ArrayList<>(); } int v1 = scan.nextInt(), v2 = scan.nextInt(); while (v1 != -1 && v2 != -1) { Graph[v1].add(v2); Graph[v2].add(v1); v1 = scan.nextInt(); v2 = scan.nextInt(); } } static void pro() { for (int i = 1; i <= n; i++) { bfs(i); } ArrayList<Integer> res = new ArrayList<>(); for (int i = 1; i <= n; i++) { if (Cnt[i] == min) res.add(i); } sb.append(min).append(' ').append(res.size()).append('\n'); Collections.sort(res); for (int i = 0; i < res.size(); i++) { sb.append(res.get(i)).append(' '); } System.out.println(sb); } static void bfs(int x) { Queue<int[]> que = new LinkedList<>(); que.add(new int[] {x, -1, 0}); int[] depth = new int[n + 1]; while (!que.isEmpty()) { int[] cur = que.poll(); // cur, prev, depth if (depth[cur[0]] == 0 || depth[cur[0]] > cur[2]){ depth[cur[0]] = cur[2]; } for (int y: Graph[cur[0]]) { if (y == cur[1]) continue; if (depth[y] == 0 || depth[y] > (cur[2] + 1)){ que.add(new int[] {y, cur[0], cur[2] + 1}); } } } HashSet<Integer> hashSet = new HashSet<>(); for (int i = 1; i <= n; i++) { if (x == i) continue; hashSet.add(depth[i]); } Cnt[x] = hashSet.size(); min = Math.min(min, Cnt[x]); } public static void main(String[] args) throws IOException { input(); pro(); } static class FastReader { BufferedReader br; StringTokenizer st; public FastReader() { br = new BufferedReader(new InputStreamReader(System.in)); } public FastReader(String s) throws FileNotFoundException { br = new BufferedReader(new FileReader(new File(s))); } String next() { while (st == null || !st.hasMoreElements()) { try { st = new StringTokenizer(br.readLine()); } catch (IOException e) { e.printStackTrace(); } } return st.nextToken(); } int nextInt() { return Integer.parseInt(next()); } long nextLong() { return Long.parseLong(next()); } double nextDouble() { return Double.parseDouble(next()); } String nextLine() { String str = ""; try { str = br.readLine(); } catch (IOException e) { e.printStackTrace(); } return str; } } }
반응형'코딩테스트 > 백준_Java' 카테고리의 다른 글
Q. 14940 자바 : 쉬운 최단거리 (0) 2023.04.22 Q. 4179 자바 : 불! (0) 2023.04.21 Q. 12427 자바 : 회사 문화 1 (0) 2023.04.19 Q. 15681 자바 : 트리와 쿼리 (0) 2023.04.17 Q. 3584 자바 : 가장 가까운 공통 조상 (0) 2023.04.16