[백준] 20924번 - 트리의 기둥과 가지 Python, Java

Tuna·2022년 3월 30일
0

Tree

목록 보기
12/13

문제


시청 공무원 마이크로는 과장으로부터 시에 있는 나무의 기둥의 길이와 가장 긴 가지의 길이를 파악하라는 업무 지시를 받았다.

마이크로는 ICPC Sinchon Winter Algorithm Camp에서 배운 트리 자료 구조를 이용하면 이 작업을 좀 더 수월하게 할 수 있으리라 판단했다.

마이크로는 트리의 기둥과 가지를 분류하기 위해 기가 노드를 추가로 정의하였다.

기가 노드는 루트 노드에서 순회를 시작했을 때, 처음으로 자식 노드가 22개 이상인 노드다. 기둥-가지를 줄여 기가 노드라 이름 붙였다. 위 그림에서 기가 노드는 44번 노드다.

단, 위 그림과 같이 리프 노드가 단 11개인 경우 리프 노드가 동시에 기가 노드가 된다.

또한, 위 그림과 같이 루트 노드가 동시에 기가 노드인 경우도 가능하다.

  • 트리의 기둥은 루트 노드에서부터 기가 노드까지다. 위 그림에서 기둥은 12341-2-3-4 이다.
    기둥의 길이는 기둥의 간선 길이의 합인 1+2+3=61 + 2 + 3 = 6 이 된다.

  • 트리의 가지는 기가 노드에서부터 임의의 리프 노드까지다. 위 그림에서 가지는 45674-5-6-7, 4584-5-8, 494-9, 410114-10-11, 410124-10-1255개가 있다.
    가지의 길이는 가지의 간선 길이의 합이다. 다행히도 가장 긴 가지의 길이 하나만 기재하면 된다. 410124-10-12 가지가 간선 길이의 합 3+3=63 + 3 = 6 으로 가장 긴 가지이다.

마이크로는 시의 나무를 트리 자료 구조로 옮겼다. 그런데 과장이 마이크로에게 또 다른 업무를 지시했다! 너무 바쁜 마이크로를 대신해 트리의 기둥과 가장 긴 가지의 길이를 측정하자.

입력


첫 번째 줄에는 노드의 개수 NN(1N2000001 \le N \le 200\,000)과 루트 노드의 번호 RR(1RN1 \le R \le N)이 주어진다.

이후 N1N-1개의 줄에 세 개의 정수 aa, bb, dd(1a,bN1 \le a, b \le N, aba \ne b)가 주어진다. 이는 aa번 노드와 bb번 노드가 연결되어있으며 이 간선의 길이가 dd(1d10001 \le d \le 1\,000)임을 의미한다. 노드는 11번부터 NN번까지 정수 번호가 매겨져 있으며 같은 간선은 여러 번 주어지지 않는다.

트리가 아닌 그래프는 입력으로 주어지지 않는다.

출력


나무의 기둥의 길이와 가장 긴 가지의 길이를 출력한다.

예제 입력 1


12 1
1 2 1
2 3 2
3 4 3
4 5 1
5 6 2
6 7 1
5 8 1
4 9 2
4 10 3
10 11 1
10 12 3

예제 출력 1


6 6

나머지 출력은 생략한다.

풀이


Python

import sys
input = sys.stdin.readline
sys.setrecursionlimit(10**9)

n,r = map(int,input().rstrip().split())
graph = [[] for _ in range(n+1)]

for _ in range(n-1):
    a,b,c = map(int,input().rstrip().split())
    graph[a].append((b,c))
    graph[b].append((a,c))


ans =[0,0]

def dfs(u,p,sum,flag):
    if flag == 0: ans[0] = sum # 기가 노드까지의 간선 길이의 합
    else: ans[1] = max(ans[1],sum) # 긴 가지의 길이
    # root가 아닌데 인접노드가 2개보다 많으면, 자식이 두개 -> giga 
    # root인데 인접 노드가 2개 이상이면 ->  giga
    if flag==0 and len(graph[u]) > 2 -int(u==r): 
        flag,sum = 1, 0
    for v,w in graph[u]:
        if v== p: continue
        dfs(v,u,sum+w,flag)

dfs(r,r,0,0)

print(ans[0],ans[1])

Java

// 추후 작성 예정

정리


  • flag를 통해서 0이면 기가 노드 까지의 간선 길이의 합을 구해주고, 1이면 리프노드를 돌면서 가장 긴 가지의 길이를 구한다.
  • 이 부분은 root가 아닌데 인접노드가 2개보다 많으면, 자식이 두개 -> giga
    root인데 인접 노드가 2개 이상이면 -> giga를 의미한다.
    if flag==0 and len(graph[u]) > 2 -int(u==r):
profile
BE 개발자가 되기 위해 노력하는 사람

0개의 댓글