ProgramSoliving

백준 : 14466 소가 길을 건넌 이유

하이후에호 2021. 4. 12. 23:11
반응형

www.acmicpc.net/problem/14466

 

14466번: 소가 길을 건너간 이유 6

첫 줄에 N, K, R이 주어진다. 다음 R줄에는 한 줄에 하나씩 길이 주어진다. 길은 상하좌우로 인접한 두 목초지를 잇고, r c r′ c′의 형태 (행, 열, 행, 열)로 주어진다. 각 수는 1 이상 N 이하이다.

www.acmicpc.net

문제유형 : 다익스트라 , 플로이드와샬

 

O(NM)이기때문에 다익스트라가 효율성이 더좋을거 같다.

 

문제의 아이디어는 소가잇는길을 가중치1 그이외에는 0으로 두고 다익스트라를 구한다.

그리고 소가 하나의 엣지에서 다른엣지로 가는길이 0이아닌 값이 무조건 다리를 건너야하는 경우다.

각각의 엣지에서 0이 아닌값을 구하면 a,b쌍 ,b,a 쌍이 나오므로 나머지 2를 해주면 정답이 된다.

package boj;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.LinkedList;
import java.util.List;
import java.util.PriorityQueue;
import java.util.Queue;
import java.util.StringTokenizer;

public class boj14466 {
	static int N, K, R;
	static boolean edge[][];
	static boolean cow[][];

	static class Edge implements Comparable<Edge> {
		int y;
		int x;
		int w;

		public Edge(int y, int x) {
			super();
			this.y = y;
			this.x = x;
		}

		public Edge(int y, int x, int w) {
			super();
			this.y = y;
			this.x = x;
			this.w = w;
		}

		@Override
		public int compareTo(Edge o) {
			return this.w - o.w;
		}
	}

	static int dy[] = { -1, 0, 1, 0 };
	static int dx[] = { 0, 1, 0, -1 };

	public static void main(String[] args) throws IOException {
		BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
		StringTokenizer st = new StringTokenizer(br.readLine());

		N = Integer.parseInt(st.nextToken());
		K = Integer.parseInt(st.nextToken());
		R = Integer.parseInt(st.nextToken());

		edge = new boolean[100001][4];
		cow = new boolean[N][N];

		for (int r = 0; r < R; r++) {
			st = new StringTokenizer(br.readLine());
			int y1 = Integer.parseInt(st.nextToken()) - 1;
			int x1 = Integer.parseInt(st.nextToken()) - 1;
			int y2 = Integer.parseInt(st.nextToken()) - 1;
			int x2 = Integer.parseInt(st.nextToken()) - 1;
			int num1 = y1 * N + x1;
			int num2 = y2 * N + x2;

			// 0 1 2 3 위 오른쪽 아래 왼쪽

			// 1을 기준으로 설계
			// 오른쪽
			if (y1 == y2 && x1 < x2) {
				edge[num1][1] = edge[num2][3] = true;
			}
			// 왼쪽
			else if (y1 == y2 && x1 > x2) {
				edge[num1][3] = edge[num2][1] = true;
			}
			// 위쪽
			else if (y1 > y2 && x1 == x2) {
				edge[num1][0] = edge[num2][2] = true;
			}
			// 아래쪽
			else {
				edge[num1][2] = edge[num2][0] = true;
			}
		}

		list = new LinkedList<>();
		for (int k = 0; k < K; k++) {
			st = new StringTokenizer(br.readLine());
			int y = Integer.parseInt(st.nextToken()) - 1;
			int x = Integer.parseInt(st.nextToken()) - 1;
			list.add(new Edge(y, x));
		}

		System.out.println(Solve());
	}

	static List<Edge> list;

	private static int Solve() {
		int cnt = 0;
		for (Edge cur : list) {
			cnt += Dickstra(cur.y, cur.x);
		}
		return cnt / 2;
	}

	static final int INF = 987654321;

	private static int Dickstra(int y, int x) {
		int[][] digit = new int[N][N];
		for (int i = 0; i < N; i++) {
			for (int j = 0; j < N; j++) {
				digit[i][j] = INF;
				digit[j][i] = INF;
			}
		}

		digit[y][x] = 0;
		Queue<Edge> q = new PriorityQueue<>();
		q.add(new Edge(y, x, 0));
		while (!q.isEmpty()) {
			Edge cur = q.poll();
			for (int d = 0; d < 4; d++) {
				int ny = cur.y + dy[d];
				int nx = cur.x + dx[d];

				if (ny < 0 || nx < 0 || nx >= N || ny >= N)
					continue;

				int num = cur.y * N + cur.x;
				// 다리있으면
				int weight = 0;

				if (edge[num][d]) {
					weight = 1;
				}

				if (digit[cur.y][cur.x] + weight < digit[ny][nx]) {
					int nw = digit[ny][nx] = digit[cur.y][cur.x] + weight;
					q.add(new Edge(ny, nx, nw));
				}
			}
		}

		int cnt = 0;
		for (Edge l : list) {
			if (digit[l.y][l.x] != 0)
				cnt++;
		}
		return cnt;
	}
}
반응형