알고리즘/[ Baekjoon ]

[ BOJ ][JAVA][12896] 스크루지 민호

kim.svadoz 2021. 5. 4. 21:51
반응형

www.acmicpc.net/problem/12896

 

12896번: 스크루지 민호

구두쇠로 유명한 스크루지 민호가 다스리는 천나라가 있다. 천나라에는 N 개의 도시들이 있는데 각각의 도시들 사이에는 양방향 도로로 이어져 있다. 민호는 도시를 세울 때 최소한의 비용만을

www.acmicpc.net

시간 제한 메모리 제한 제출 정답 맞은 사람 정답 비율
2 초 512 MB 88 33 27 38.028%

문제

구두쇠로 유명한 스크루지 민호가 다스리는 천나라가 있다. 천나라에는 N 개의 도시들이 있는데 각각의 도시들 사이에는 양방향 도로로 이어져 있다. 민호는 도시를 세울 때 최소한의 비용만을 들이고 싶어서 N - 1 개의 도로를 이용해 모든 도시들 사이에는 단 한개의 경로만이 존재하도록 도시를 세웠다.

도시를 세울 당시에 소방서를 여러개 건설하는 것이 아까웠던 스쿠르지 민호는 단 하나의 도시에 소방서를 건설하기로 했다. 하지만 최소한의 양심이 있어서인지 소방서는 최적의 위치가 될 수 있는 도시에 건설하기로 했다. 최적의 위치라는 것은 소방서에서 소방차가 출동해 다른 도시에 도착을 할 때 이동해야 하는 거리중의 최대가 최소가 되는 지점을 의미한다. 편의상 같은 도시 내에서 이동하는 거리는 없다고 생각하며 한 도시에서 다른 도시로 연결된 도로는 거리가 1이라고 생각한다.

천나라에 있는 도시의 수와 도로들의 연결 상태가 주어질 때 최적의 위치에 설치된 소방서에서 소방차가 출동해 다른 도시에 도착할 때 이동해야하는 거리들 중 최대 거리를 구하는 프로그램을 작성하자.

입력

첫째 줄에는 천나라에 있는 도시의 수 N (2 <= N <= 100,000) 이 주어진다. 다음 N - 1 줄에 걸쳐 도시들의 연결 상태가 주어진다.

각각의 줄에는 공백을 기준으로 세개의 숫자가 u, v (1 <= u, v <= N) 가 주어지는데 이는 도시 u와 v가 양방향 도로로 연결이 되어 있다는 것을 의미한다.

출력

첫째 줄에 최적의 위치에 설치된 소방서에서 소방차가 출동해 다른 도시에 도착할 때까지 이동해야하는 거리들 중 최댓값을 출력한다.

예제 입력 1

5
4 5
4 2
2 3
1 2

예제 출력 1

2

코드

import java.util.*;
import java.io.*;

public class p12896 {
    static class Pair {
        int v, cost;
        public Pair(int v, int cost) {
            this.v = v;
            this.cost = cost;
        }
    }
    static int n;
    static List<Pair>[] list;
    static int[] dp;
    static boolean[] visit;
    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        StringTokenizer st;
        n = Integer.parseInt(br.readLine());
        // 최대 도시의 수보다 넉넉히 할당
        list = new ArrayList[100005];
        dp = new int[100005];
        visit = new boolean[100005];
        for (int i = 0; i <= n; i++) {
            list[i] = new ArrayList<>();
        }
        for (int i = 0; i < n - 1; i++) {
            st = new StringTokenizer(br.readLine());
            int u = Integer.parseInt(st.nextToken());
            int v = Integer.parseInt(st.nextToken());
            list[u].add(new Pair(v, 1));
            list[v].add(new Pair(u, 1));
        }

        // 가장 길이(cost)가 긴 자식을 리턴
        Pair p = dfs(1);
        visit = new boolean[n + 1];
        // 가장 길이가 긴 자식에서 다시 dfs를 돈다.
        Pair r = dfs(p.v);
        // 가장 먼 정점과의 거리 중 최소값
        System.out.println((1 + r.cost) / 2);
    }


    static Pair dfs(int cur) {
        Pair result = new Pair(cur, 0);
        // 노드 방문처리
        visit[cur] = true;
        for (Pair next : list[cur]) {
            // 자식 노드를 방문했다면 넘어가고
            if (visit[next.v]) continue;
            // 방문하지 않았다면 즉시 방문해본다.
            Pair x = dfs(next.v);
            // 모든 자식까지 모두 방문이 끝나면 리프노드부터 부모노드의 비용을 더한다.
            x.cost += next.cost;
            // 항상 자식노드들 중 가장 큰 비용을 가진 자식을 리턴한다.
            if (x.cost > result.cost) {
                result = x;
            }
        }

        return result;
    }
}
반응형