mirror of
https://github.com/TheAlgorithms/JavaScript.git
synced 2025-07-05 16:26:47 +08:00
127 lines
3.7 KiB
JavaScript
127 lines
3.7 KiB
JavaScript
/*
|
|
* Problem Statement:
|
|
* - Given a NxN grid, find whether rat in cell [0, 0] can reach the target in cell [N-1, N-1]
|
|
* - The grid is represented as an array of rows. Each row is represented as an array of 0 or 1 values.
|
|
* - A cell with value 0 can not be moved through. Value 1 means the rat can move here.
|
|
* - The rat can not move diagonally.
|
|
*
|
|
* Reference for this problem: https://www.geeksforgeeks.org/rat-in-a-maze-backtracking-2/
|
|
*
|
|
* Based on the original implementation contributed by Chiranjeev Thapliyal (https://github.com/chiranjeev-thapliyal).
|
|
*/
|
|
|
|
/**
|
|
* Checks if the given grid is valid.
|
|
*
|
|
* A grid needs to satisfy these conditions:
|
|
* - must not be empty
|
|
* - must be a square
|
|
* - must not contain values other than {@code 0} and {@code 1}
|
|
*
|
|
* @param grid The grid to check.
|
|
* @throws TypeError When the given grid is invalid.
|
|
*/
|
|
function validateGrid (grid) {
|
|
if (!Array.isArray(grid) || grid.length === 0) throw new TypeError('Grid must be a non-empty array')
|
|
|
|
const allRowsHaveCorrectLength = grid.every(row => row.length === grid.length)
|
|
if (!allRowsHaveCorrectLength) throw new TypeError('Grid must be a square')
|
|
|
|
const allCellsHaveValidValues = grid.every(row => {
|
|
return row.every(cell => cell === 0 || cell === 1)
|
|
})
|
|
if (!allCellsHaveValidValues) throw new TypeError('Grid must only contain 0s and 1s')
|
|
}
|
|
|
|
function isSafe (grid, x, y) {
|
|
const n = grid.length
|
|
return x >= 0 && x < n && y >= 0 && y < n && grid[y][x] === 1
|
|
}
|
|
|
|
/**
|
|
* Attempts to calculate the remaining path to the target.
|
|
*
|
|
* @param grid The full grid.
|
|
* @param x The current X coordinate.
|
|
* @param y The current Y coordinate.
|
|
* @param solution The current solution matrix.
|
|
* @param path The path we took to get from the source cell to the current location.
|
|
* @returns {string|boolean} Either the path to the target cell or false.
|
|
*/
|
|
function getPathPart (grid, x, y, solution, path) {
|
|
const n = grid.length
|
|
|
|
// are we there yet?
|
|
if (x === n - 1 && y === n - 1 && grid[y][x] === 1) {
|
|
solution[y][x] = 1
|
|
return path
|
|
}
|
|
|
|
// did we step on a 0 cell or outside the grid?
|
|
if (!isSafe(grid, x, y)) return false
|
|
|
|
// are we walking onto an already-marked solution coordinate?
|
|
if (solution[y][x] === 1) return false
|
|
|
|
// none of the above? let's dig deeper!
|
|
|
|
// mark the current coordinates on the solution matrix
|
|
solution[y][x] = 1
|
|
|
|
// attempt to move right
|
|
const right = getPathPart(grid, x + 1, y, solution, path + 'R')
|
|
if (right) return right
|
|
|
|
// right didn't work: attempt to move down
|
|
const down = getPathPart(grid, x, y + 1, solution, path + 'D')
|
|
if (down) return down
|
|
|
|
// down didn't work: attempt to move up
|
|
const up = getPathPart(grid, x, y - 1, solution, path + 'U')
|
|
if (up) return up
|
|
|
|
// up didn't work: attempt to move left
|
|
const left = getPathPart(grid, x - 1, y, solution, path + 'L')
|
|
if (left) return left
|
|
|
|
// no direction was successful: remove this cell from the solution matrix and backtrack
|
|
solution[y][x] = 0
|
|
return false
|
|
}
|
|
|
|
function getPath (grid) {
|
|
// grid dimensions
|
|
const n = grid.length
|
|
|
|
// prepare solution matrix
|
|
const solution = []
|
|
for (let i = 0; i < n; i++) {
|
|
const row = Array(n)
|
|
row.fill(0)
|
|
solution[i] = row
|
|
}
|
|
|
|
return getPathPart(grid, 0, 0, solution, '')
|
|
}
|
|
|
|
/**
|
|
* Creates an instance of the "rat in a maze" based on a given grid (maze).
|
|
*/
|
|
export class RatInAMaze {
|
|
constructor (grid) {
|
|
// first, let's do some error checking on the input
|
|
validateGrid(grid)
|
|
|
|
// attempt to solve the maze now - all public methods only query the result state later
|
|
const solution = getPath(grid)
|
|
|
|
if (solution !== false) {
|
|
this.path = solution
|
|
this.solved = true
|
|
} else {
|
|
this.path = ''
|
|
this.solved = false
|
|
}
|
|
}
|
|
}
|