[BaekJoon] 19581 두 번째 트리의 지름 (Java)

오태호·2023년 10월 17일
0

백준 알고리즘

목록 보기
335/396
post-thumbnail

1.  문제 링크

https://www.acmicpc.net/problem/19581

2.  문제


3.  소스코드

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.*;

public class Main {
    static int nodeNum;
    static Map<Integer, List<Edge>> tree;
    static boolean[] visited;

    static int longestNode;
    static int longestDist;

    static void input() {
        Reader scanner = new Reader();

        nodeNum = scanner.nextInt();
        tree = new HashMap<>();
        for(int node = 1; node <= nodeNum; node++) {
            tree.put(node, new ArrayList<>());
        }

        for(int edge = 0; edge < nodeNum - 1; edge++) {
            int node1 = scanner.nextInt();
            int node2 = scanner.nextInt();
            int weight = scanner.nextInt();

            tree.get(node1).add(new Edge(node2, weight));
            tree.get(node2).add(new Edge(node1, weight));
        }
    }

    /*
        두 번쨰 트리의 지름은 트리의 지름을 이루는 두 정점 각각에서 상대방을 제외한 가장 긴 정점까지의 거리가 될 것이다
        두 개의 거리가 나올 수 있으니 그 두 거리 중 더 긴 거리가 두 번째 트리의 지름이 될 것이다

        지름을 이루는 두 정점 구하기
            - 아무 정점에서 가장 긴 정점을 구하면 지름을 이루는 두 정점 중 하나의 정점을 구할 수 있다
            - 지름을 이루는 두 정점 중 하나의 정점을 구했으니 그 정점에서 가장 긴 다른 정점을 구하면 된다

        두 번째 트리의 지름 구하기
            - 지름을 이루는 두 정점을 구하는 것처럼 지름을 이루는 두 정점으로부터 가장 긴 정점을 각각 구한다
            - 이때 상대방 정점을 제외한 가장 긴 정점을 구한다
     */
    static void solution() {
        int nodeOfDiameter1 = getNodeOfDiameter(1);
        int nodeOfDiameter2 = getNodeOfDiameter(nodeOfDiameter1);

        int longestDist1 = getLongestDistExceptOneNode(nodeOfDiameter1, nodeOfDiameter2);
        int longestDist2 = getLongestDistExceptOneNode(nodeOfDiameter2, nodeOfDiameter1);

        System.out.println(Math.max(longestDist1, longestDist2));
    }

    static int getLongestDistExceptOneNode(int node, int exceptNode) {
        longestNode = longestDist = 0;
        visited = new boolean[nodeNum + 1];
        dfs(node, 0, exceptNode);

        return longestDist;
    }

    static int getNodeOfDiameter(int node) {
        longestNode = longestDist = 0;
        visited = new boolean[nodeNum + 1];
        dfs(node, 0, 0);

        return longestNode;
    }

    static void dfs(int node, int weight, int exceptNode) {
        if(weight > longestDist) {
            longestNode = node;
            longestDist = weight;
        }

        visited[node] = true;

        for(Edge next : tree.get(node)) {
            if(next.node == exceptNode) {
                continue;
            }

            if(!visited[next.node]) {
                dfs(next.node, weight + next.weight, exceptNode);
            }
        }
    }

    static class Edge {
        int node;
        int weight;

        public Edge(int node, int weight) {
            this.node = node;
            this.weight = weight;
        }
    }

    public static void main(String[] args) {
        input();
        solution();
    }

    static class Reader {
        BufferedReader br;
        StringTokenizer st;

        public Reader() {
            br = new BufferedReader(new InputStreamReader(System.in));
        }

        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());
        }
    }
}
profile
자바, 웹 개발을 열심히 공부하고 있습니다!

0개의 댓글