[프로그래머스] 트리 트리오 중간값

donghyeok·2023년 4월 20일
0

알고리즘 문제풀이

목록 보기
115/171

문제 설명

https://school.programmers.co.kr/learn/courses/30/lessons/68937

문제 풀이

  • 트리, BFS, 중간값에 대한 이해가 필요한 문제이다.
  • 중간값이란 세개의 수를 정렬했을 때 2번째 수를 뜻한다 (!=평균값)
  • 트리의 중요한 특징은 다음과 같다.
    1. 사이클이 존재하지 않음
    2. 특정 노드에서 특정 노드로 가는 경로는 1가지 뿐이다.
  • 문제를 풀기 위해서는 트리의 지름(노드 간의 거리가 가장 먼 두 노드 사이 거리)를 구해야 한다. 방법은 다음과 같다.
    1. 특정 노드에서 BFS로 가장 먼 leaf 노드를 구한다. (지름은 2개의 leaf 사이의 거리임)
    2. 해당 leaf 노드에서 가장 먼 leaf 노드 사이의 거리를 구한다.
  • 위처럼 지름이 구해졌으면 문제의 답은 아래 2가지 중 하나이다.
    1. 지름의 양끝 노드에서 지름만큼의 거리를 가지는 노드가 2개 이상 존재한다 -> 중간값 : 지름
    2. 지름의 양끝 노드에서 지름만큼의 거리를 가지는 노드가 1개만 존재한다. -> 중간값 : 지름 - 1
  • 자세한 구현은 코드를 참조

소스 코드 (JAVA)

import java.util.*;

class Solution {
    public class ResultData {
        public List<Integer> list;
        public int depth;

        public ResultData(List<Integer> list, int depth) {
            this.list = list;
            this.depth = depth;
        }
    }
    public List<List<Integer>> map = new ArrayList<>();

    public ResultData bfs(int leaf, int n) {
        boolean[] chk = new boolean[n + 1];
        Queue<Integer> q1 = new ArrayDeque<>();
        Queue<Integer> q2 = new ArrayDeque<>();

        q1.add(leaf);
        chk[leaf] = true;
        int depth = 0;
        List<Integer> list = new ArrayList<>();
        while(true) {
            //BFS 진행
            while (!q1.isEmpty()) {
                int cur = q1.poll();
                for (int next : map.get(cur)) {
                    if (chk[next]) continue;
                    chk[next] = true;
                    q2.add(next);
                }
            }

            if (q2.isEmpty())
                break;

            //결과 계산
            depth++;
            list = new ArrayList<>(q2);

            //q2 -> q1 옮겨줌
            while(!q2.isEmpty()) {
                q1.add(q2.poll());
            }
        }
        return new ResultData(list, depth);
    }

    public int solution(int n, int[][] edges) {
        //초기화
        for (int i = 0; i <= n; i++)
            map.add(new ArrayList<>());
        for (int i = 0; i < edges.length; i++) {
            int from = edges[i][0];
            int to = edges[i][1];
            map.get(from).add(to);
            map.get(to).add(from);
        }

        //1. BFS를 통해서 1번에서 가장 먼 leaf 노드 구하기
        int cand = 0;
        Queue<Integer> q = new ArrayDeque<>();
        boolean[] chk = new boolean[n+1];

        q.add(1);
        chk[1] = true;
        while(!q.isEmpty()) {
            int cur = q.poll();
            if (map.get(cur).size() == 1)
                cand = cur;
            for (int next : map.get(cur)) {
                if (chk[next]) continue;
                chk[next] = true;
                q.add(next);
            }
        }

        //2. 해당 leaf 노드에서 BFS 진행하여 최대 거리 및 최대 거리 개수를 구한다.
        ResultData result1 = bfs(cand, n);
        if (result1.list.size() > 1)
            return result1.depth;

        //3. 최대 거리 개수가 1개이면 해당 노드에서 다시 BFS 진행
        ResultData result2 = bfs(result1.list.get(0), n);
        if (result2.list.size() > 1)
            return result2.depth;
        else
            return result2.depth-1;
    }
}

0개의 댓글