[BOJ 1504] 특정한 최단경로 (Java)

nnm·2020년 2월 27일
0
post-custom-banner

BOJ 1504 특정한 최단경로

문제풀이

이 문제는 최단경로를 구하는 문제지만 다음과 같은 특징을 가지고 있다.

  • 방문했던 정점, 간선을 재방문 할 수 있다.
    방문체크를 하지않는다.
  • 특정한 두 정점을 반드시 지나서 도착해야한다.
    시작점 - A - B - 도착점 , 시작점 - B - A - 도착점 이라는 두 가지 경우가 생긴다.

따라서 시작점, A, B를 시작점으로 다익스트라를 총 3번 수행하여 각 경우에 맞게 합하여 비교해서 최단 거리를 구하면 된다.

구현코드

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.PriorityQueue;
import java.util.StringTokenizer;

public class Main {
	
	static class Edge implements Comparable<Edge>{
		int to, weight;
		
		Edge(int to, int weight){
			this.to = to;
			this.weight = weight;
		}

		@Override
		public int compareTo(Edge o) {
			return this.weight - o.weight;
		}
	}
	
	static PriorityQueue<Edge> pq;
	static ArrayList<ArrayList<Edge>> adj;
	static int[][] dist;
	static int N, E, first, second;

	public static void main(String[] args) throws IOException {
		BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
		StringTokenizer st = new StringTokenizer(br.readLine());
		
		N = stoi(st.nextToken());
		E = stoi(st.nextToken());
		
		pq = new PriorityQueue<>();
		dist = new int[3][N + 1];
		adj = new ArrayList<>();
		
		for(int i = 0 ; i < 3 ; ++i) {
			Arrays.fill(dist[i], Integer.MAX_VALUE);
		}
		
		for(int i = 0 ; i <= N ; ++i) {
			adj.add(new ArrayList<Edge>());
		}
		
		// 인접리스트 만들기 
		for(int i = 0 ; i < E ; ++i) {
			st = new StringTokenizer(br.readLine());
			int from = stoi(st.nextToken());
			int to = stoi(st.nextToken());
			int weight = stoi(st.nextToken());
			
			adj.get(to).add(new Edge(from, weight));
			adj.get(from).add(new Edge(to, weight));
		}
		
		st = new StringTokenizer(br.readLine());
		first = stoi(st.nextToken());
		second = stoi(st.nextToken());
		
		getMinimumDistance(1, 0);
		getMinimumDistance(first, 1);
		getMinimumDistance(second, 2);
		
		if(dist[0][N] == Integer.MAX_VALUE) {
			System.out.println(-1);
			return;
		}
		
		int way1 = dist[0][first] + dist[1][second] + dist[2][N];
		int way2 = dist[0][second] + dist[2][first] + dist[1][N];
		
		if(way1 > way2) {
			System.out.println(way2);
		} else {
			System.out.println(way1);
		}
	}
	
	private static void getMinimumDistance(int start, int i) {
		dist[i][start] = 0;
		pq.offer(new Edge(start, 0));
		dijkstra(i);
	}
	
	private static void dijkstra(int i) {
		while(!pq.isEmpty()) {
			Edge e = pq.poll();
			
			for(Edge ne : adj.get(e.to)) {
				if(dist[i][ne.to] > dist[i][e.to] + ne.weight) {
					dist[i][ne.to] = dist[i][e.to] + ne.weight;
					pq.offer(ne);
				}
			}
		}
	}

	private static int stoi(String s) {
		return Integer.parseInt(s);
	}
}
profile
그냥 개발자
post-custom-banner

0개의 댓글