본문 바로가기
백준

백준 1368번 : 물대기 (Java)

by Huiyeong 2022. 2. 14.

 문제

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

 

1368번: 물대기

첫 줄에는 논의 수 N(1 ≤ N ≤ 300)이 주어진다. 다음 N개의 줄에는 i번째 논에 우물을 팔 때 드는 비용 Wi(1 ≤ Wi ≤ 100,000)가 순서대로 들어온다. 다음 N개의 줄에 대해서는 각 줄에 N개의 수가 들어

www.acmicpc.net

1368 물대기

 

 구현 방법

최소의 비용으로 모든 논에 물을 대야하기 때문에 Kruskal 알고리즘을 사용했습니다.

물을 대는 방법은 직접 논에 우물 파는 방법, 이미 물을 대고 있는 다른 논으로부터 물을 끌어오는 방법 2개가 있어 적절한 방법을 골라 사용해야 합니다.

처음에는 너무 복잡하게 생각했는데 사실 우물이라는 가상의 노드를 추가하면 해결되는 문제였습니다.

즉, 우물이라는 가상의 노드 (0)를 추가해주어 우물을 포함한 모든 노드를 연결시켜 주면 끝!

 

 구현 코드

package BOJ.Gold;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.StringTokenizer;

public class B1368_물대기 {
	static int N, root[], ans, cnt;
	static List<Edge> edgeList = new ArrayList<>();
	static class Edge implements Comparable<Edge>{
		int from, to, weight;

		public Edge(int from, int to, int weight) {
			super();
			this.from = from;
			this.to = to;
			this.weight = weight;
		}

		@Override
		public int compareTo(Edge o) {
			return Integer.compare(this.weight, o.weight);
		}
		
	}
	public static void main(String[] args) throws NumberFormatException, IOException {
		BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
		N = Integer.parseInt(br.readLine());
		root = new int[N+1];
		for (int i=1;i<=N;i++) {
			// 우물이라는 가상의 노드 (0)가 있다고 가정하여 간선 추가
			edgeList.add(new Edge(0, i, Integer.parseInt(br.readLine())));
		}
		StringTokenizer st = null;
		for (int i=1;i<=N;i++) {
			st = new StringTokenizer(br.readLine());
			for (int j=1;j<=N;j++) {
				int value = Integer.parseInt(st.nextToken());
				if (i!=j) edgeList.add(new Edge(i, j, value)); // 간선 추가
			}
		}
		makeSet(); 
		Collections.sort(edgeList);
		for (Edge edge : edgeList) {
			if (union(edge.from, edge.to)) { // 연결이 안되어 있으면 연결 시켜주고
				ans += edge.weight; // 비용 추가
				if (++cnt == N) { // 우물 포함 모든 노드가 연결 되면 끝
					System.out.println(ans);
					System.exit(0);
				}
			}
		}
	}
	
	static void makeSet() {
		for (int i=0;i<=N;i++) {
			root[i] = -1;
		}
	}
	
	static int findSet(int x) {
		if (root[x] < 0) return x;
		return root[x] = findSet(root[x]);
	}
	
	static boolean union(int a, int b) {
		int aRoot = findSet(a);
		int bRoot = findSet(b);
		if (aRoot == bRoot) return false; // 이미 연결되어있음
		root[bRoot] = aRoot; // 연결 시켜줌
		return true;
	}
}

 

정답!