티스토리 뷰

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

 

13459번: 구슬 탈출

첫 번째 줄에는 보드의 세로, 가로 크기를 의미하는 두 정수 N, M (3 ≤ N, M ≤ 10)이 주어진다. 다음 N개의 줄에 보드의 모양을 나타내는 길이 M의 문자열이 주어진다. 이 문자열은 '.', '#', 'O', 'R', 'B'

www.acmicpc.net

 

// https://www.acmicpc.net/problem/13459
// 구슬 탈출

class Item {
	next = null;
	constructor(value) {
		this.value = value;
	}
}

class Queue {
	head = null;
	tail = null;
	size = 0;
	constructor() {}

	push(value) {
		const node = new Item(value);
		if (this.head == null) {
			this.head = node;
		} else if (this.tail) {
			this.tail.next = node;
		}
		this.tail = node;
		this.size += 1;
	}

	pop() {
		if (this.head) {
			const popItem = this.head;
			this.head = this.head.next;
			this.size -= 1;
			return popItem.value;
		} else {
			return null;
		}
	}
}

const input = require('fs').readFileSync('./dev/stdin').toString().trim().split('\n');

const [R, C] = input.shift().split(' ').map(Number);
const board = input.map((v) => v.split(''));
let redX;
let redY;
let blueX;
let blueY;
let outX;
let outY;

for (let i = 0; i < R; i++) {
	for (let j = 0; j < C; j++) {
		if (board[i][j] == 'R') {
			redX = i;
			redY = j;
			board[i][j] = '.';
		}
		if (board[i][j] == 'B') {
			blueX = i;
			blueY = j;
			board[i][j] = '.';
		}
		if (board[i][j] == 'O') {
			outX = i;
			outY = j;
		}
	}
}

const dx = [0, 0, 1, -1];
const dy = [1, -1, 0, 0];
const q = new Queue();
let answer = 0;
q.push([redX, redY, blueX, blueY, 0]);

while (q.size > 0 && answer == 0) {
	const [orx, ory, obx, oby, c] = q.pop();
	for (let i = 0; i < 4; i++) {
		let rx = orx;
		let ry = ory;
		let bx = obx;
		let by = oby;
		let redEnd = false;
		let blueEnd = false;
		while (!(redEnd && blueEnd)) {
			let nrx = rx;
			let nry = ry;
			let nbx = bx;
			let nby = by;
			if (!redEnd) {
				nrx += dx[i];
				nry += dy[i];
			}
			if (!blueEnd) {
				nbx += dx[i];
				nby += dy[i];
			}

			if (
				board[nrx][nry] == '.' &&
				!(board[bx + dx[i]][by + dy[i]] == '#' && nrx == bx && nry == by)
			) {
				// 새로운 빨강공이 갈수 있는 위치면
				//  벽이 아니고
				//  해당 위치에 파란공이 없거나
				//  파란공이 있어도 파란공 역시 이동할 수 있는 경우
				rx = nrx;
				ry = nry;
			} else if (board[nrx][nry] == 'O') {
				rx = nrx;
				ry = nry;
				redEnd = true;
			} else {
				redEnd = true;
			}

			if (
				board[nbx][nby] == '.' &&
				!(board[rx + dx[i]][ry + dy[i]] == '#' && nbx == rx && nby == ry)
			) {
				bx = nbx;
				by = nby;
			} else if (board[nbx][nby] == 'O') {
				bx = nbx;
				by = nby;
				blueEnd = true;
			} else {
				blueEnd = true;
			}
		}

		if (board[bx][by] == 'O' || (rx == orx && ry == ory && bx == obx && by == oby)) {
			continue;
		} else if (board[rx][ry] == 'O' && board[bx][by] != 'O') {
			answer = 1;
			break;
		} else {
			if (c < 9) q.push([rx, ry, bx, by, c + 1]);
		}
	}
}

console.log(answer);
728x90
공지사항
최근에 올라온 글
최근에 달린 댓글
Total
Today
Yesterday
링크
«   2025/01   »
1 2 3 4
5 6 7 8 9 10 11
12 13 14 15 16 17 18
19 20 21 22 23 24 25
26 27 28 29 30 31
글 보관함