본문 바로가기
개발/코딩테스트

백준 1197번: 최소 스패닝 트리 [JAVA]

by 로또 2023. 3. 27.

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

 

1197번: 최소 스패닝 트리

첫째 줄에 정점의 개수 V(1 ≤ V ≤ 10,000)와 간선의 개수 E(1 ≤ E ≤ 100,000)가 주어진다. 다음 E개의 줄에는 각 간선에 대한 정보를 나타내는 세 정수 A, B, C가 주어진다. 이는 A번 정점과 B번 정점이

www.acmicpc.net

  • 풀이

최소 신장 트리는 크루스칼 알고리즘과 프림 알고리즘으로 해결할 수 있다.

나는 그 중 크루스칼 알고리즘을 풀었다.

Edge 클래스를 만들어 각 간선의 정점, 가중치 값을 담고 가중치 오름차순으로 정렬했다.

가중치가 작은 것부터 시작해서 간선이 잇는 두 정점이 같은 그룹이 아니라면 선택하는 방식이다. 두 정점이 이미 같은 그룹이라면 다시 선택했을 때 사이클이 형성되기 때문이다.

두 정점이 같은 그룹인 지 확인하기 위해 부모를 찾는 Find 함수를 사용하였고, 간선이 선택되었을 때 같은 그룹으로 만들어주기 위해 부모를 합치는 Union 함수를 사용하였다.

  • 코드
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.Arrays;
import java.util.StringTokenizer;

public class Main {
	static int V, E;
	static int parent[];
	static Edge edges[];
	static int weight;
	
	static class Edge{
		int u, v, c;
		Edge(int u, int v, int c){
			this.u = u;
			this.v = v;
			this.c = c;
		}
	}
	
	// 같은 그룹인 지 확인
	static int find(int x) {
		if (parent[x] == x) return x;
		return parent[x] = find(parent[x]);
	}
	
	// 같은 그룹으로 합치기
	static void union(int x, int y) {
		x = parent[x];
		y = parent[y];
		
		if (x>y) parent[x] = y;
		else parent[y] = x;
	}
	

	public static void main(String[] args) throws IOException {
		BufferedReader bf = new BufferedReader(new InputStreamReader(System.in));
		StringTokenizer st = new StringTokenizer(bf.readLine());
		
		V = Integer.parseInt(st.nextToken());
		E = Integer.parseInt(st.nextToken());

		parent = new int[V+1];
		edges = new Edge[E];
		
	  // 부모 배열 초기화
		for (int i=1; i<V+1; i++) parent[i] = i;
		
		// 간선 정보 입력
		for (int i=0; i<E; i++) {
			st = new StringTokenizer(bf.readLine());
			int u = Integer.parseInt(st.nextToken());
			int v = Integer.parseInt(st.nextToken());
			int c = Integer.parseInt(st.nextToken());
			edges[i] = new Edge(u, v, c);
		}
		
		// 가중치 오름차순 정렬
		Arrays.sort(edges, (o1, o2) -> (o1.c - o2.c));
		
		// 크루스칼 알고리즘
		for (int i=0; i<E; i++) {
			Edge now = edges[i];
			if (find(now.u) == find(now.v)) continue;
			weight += now.c;
			union (now.u, now.v);
		}

		// 출력
		System.out.println(weight);
	}

}

댓글