[Java] 백준 3780: 네트워크 연결

Cyan·2026년 3월 31일

코딩 테스트

목록 보기
185/192

백준 3780: 네트워크 연결

문제 요약

종빈이는 아주 큰 그룹의 총수다. 이 그룹은 1부터 N번까지의 번호로 구분할 수 있는 N개의 기업을 운영하고 있다. 현재 각 기업은 서로 독립적인 자체 컴퓨팅 및 통신센터를 가지고 있다.

어느 날 종빈이는 계열사의 CTO인 서현이에게 서비스 개선을 위해 각 기업의 서버를 네트워크로 연결하여 단일 통신센터에서 관리 가능한 클러스터 형태로 구성할 것을 제안했다. 종빈이의 제안을 들은 서현이는 다음과 같은 병합 과정을 고안해냈다.

클러스터 A를 제공하는 기존에 존재하는 센터 I를 고른다.
클러스터 B를 제공하는 기업 J를 고른다. B는 A가 아닌 임의의 클러스터이며, J는 센터가 아닐 수 있다.
I와 J를 통신 라인으로 연결한다. 이때 기업 I와 J를 잇는 라인의 길이는 |I – J|(mod 1000)이다.
위 방식을 통해 클러스터 A와 B는 새로운 클러스터로 합쳐지며, 이 클러스터는 B의 센터에 의해 제공된다.
이러한 병합 과정을 거치던 중에, 각 기업에서 현재 센터까지 연결되는 라인의 길이가 총 얼마나 되는지에 관한 문의가 들어왔다. 서현이를 위해 병합하는 과정과 그 과정에서 통신센터와 각 기업을 잇는 라인의 길이를 구하는 프로그램을 작성해보자.

문제 분류

  • 자료 구조
  • 분리 집합

문제 풀이

문제를 몇 번인가 오해했다. 거리 배열을 함께 다루는 유니온 파인드 문제이다. 우선 두 클러스터를 병합할 경우 IJ에 연결하는 연산이다. 이 때, I의 거리 dis[I]IJ의 사이 거리만큼 누적시켜 갱신한다. 어떤 정점과 그 클러스터의 센터간 거리를 출력하는 쿼리에서는 find()를 통해 해당 정점에서 센터까지의 거리를 갱신하여 출력하면 된다.

풀이 코드

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

public class Main {
	static int n;
	static int p[], dis[];
	static int find(int v) {
		if(p[v] <= 0) return v;
		find(p[v]);
		dis[v] += dis[p[v]];
		p[v] = find(p[v]);
		return p[v];
	}
	static void merge(int a, int b) {
		dis[a] += Math.abs(a - b) % 1000;
		p[a] = b;
	}
	public static void main(String[] args) throws Exception {
		BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
		StringTokenizer st;
		StringBuilder sb = new StringBuilder();
		
		String com;
		int T = Integer.parseInt(br.readLine());
		for(int tc = 1; tc <= T; tc++) {
			n = Integer.parseInt(br.readLine());
			p = new int[n + 1];
			dis = new int[n + 1];
			while(true) {
				st = new StringTokenizer(br.readLine());
				com = st.nextToken();
				if(com.equals("O")) break;
				else if(com.equals("E")) {
					int v = Integer.parseInt(st.nextToken());
					find(v);
					sb.append(dis[v]).append("\n");
				}
				else
					merge(Integer.parseInt(st.nextToken()), Integer.parseInt(st.nextToken()));
			}
		}
		System.out.println(sb);
	}
}

0개의 댓글