Created
January 12, 2022 15:29
-
-
Save windmaomao/d13f69d92a85124e9d8dad2f7b803618 to your computer and use it in GitHub Desktop.
world interpretation
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
const move = (a, b) => [a[0]+b[0], a[1]+b[1]] | |
// given data build a maze matrix | |
const buildMaze = (data) => { | |
const maze = data.slice(1).split('\n') | |
.map(r => r.split('')) | |
let startPos = [0, 0], numKeys = 0 | |
for (let i = 0; i < maze.length; i++) { | |
const row = maze[i] | |
const j = row.indexOf('@') | |
if (j >= 0) { | |
startPos = [i, j] | |
} | |
numKeys += row.filter(c => c.match(/[a-z]/)) | |
.length | |
} | |
return { maze, startPos, numKeys } | |
} | |
// add a global storage | |
function vec(n) { | |
const values = new Array(n).fill([[0,0], 0]) | |
let length = 0 | |
const len = () => length | |
const value = (i) => values[i] | |
const clear = () => { length = 0 } | |
const push = v => { | |
values[length] = v | |
length += 1 | |
if (length >= n) { | |
console.error(`memory overflow at ${length}`) | |
} | |
} | |
return { value, clear, push, len, values } | |
} | |
const queue = vec(8000) | |
// starting at a source pos with keys taken | |
// find next keys | |
const dirs = [[-1,0], [0,1], [1,0], [0,-1]] | |
const findMazeKeys = (maze, srcPos, keysTaken) => { | |
queue.clear() | |
queue.push([srcPos, 0]) | |
const marked = {} | |
const dests = {} | |
let ii = 0 | |
const canOpen = c => keysTaken | |
.indexOf(c.toLowerCase()) >= 0 | |
while (ii < queue.len()) { | |
const [pos, k] = queue.value(ii++) | |
marked[pos] = true | |
const c = maze[pos[0]][pos[1]] | |
if (c.match(/[a-z]/) | |
&& keysTaken.indexOf(c) < 0) { | |
dests[c] = { pos, k } | |
} else { | |
for (let dp of dirs) { | |
const p = move(pos, dp) | |
const pc = maze[p[0]][p[1]] | |
if (!marked[p] && (pc !== '#') && | |
!(pc.match(/[A-Z]/) && !canOpen(pc))) { | |
queue.push([p, k + 1]) | |
} | |
} | |
} | |
} | |
return dests | |
} | |
const genMemoKey = (keys, p) => | |
[...keys].sort().join('') + `:(${p})` | |
const findMazeSteps = (maze, startPos, numKeys) => { | |
// starting at a source pos with keys taken | |
// find the minimium steps to collect rest keys | |
const minMazeSteps = (srcPos, keysTaken) => { | |
const memoKey = genMemoKey(keysTaken, srcPos) | |
let res = Infinity | |
if (memo[memoKey] == undefined) { | |
if (keysTaken.length == numKeys) { | |
res = 0 | |
} else { | |
const ks = findMazeKeys(maze, srcPos, keysTaken) | |
Object.entries(ks).forEach(([c, { pos, k }]) => { | |
res = Math.min( | |
res, | |
minMazeSteps(pos, [...keysTaken, c]) + k | |
) | |
}) | |
} | |
actual += 1 | |
memo[memoKey] = res | |
} | |
usage += 1 | |
return memo[memoKey] | |
} | |
let memo = {}, usage = 0, actual = 0 | |
console.time('time') | |
const tmp = minMazeSteps(startPos, []) | |
console.log('usage:', usage, actual) | |
console.timeEnd('time') | |
return tmp | |
} | |
function World(data) { | |
const { maze, startPos, numKeys } = buildMaze(data) | |
return findMazeSteps(maze, startPos, numKeys) | |
} | |
const {log} = console | |
const world1 = ` | |
######### | |
#[email protected]# | |
#########` // 8 | |
//log(World(world1)) | |
const world2 = ` | |
######################## | |
#[email protected].# | |
######################.# | |
#d.....................# | |
########################` // 86 | |
//log(World(world2)) | |
const world3 = ` | |
################# | |
#i.G..c...e..H.p# | |
########.######## | |
#j.A..b...f..D.o# | |
########@######## | |
#k.E..a...g..B.n# | |
########.######## | |
#l.F..d...h..C.m# | |
#################` // 136 | |
//log(World(world3)) | |
const world4 = ` | |
######################## | |
#@..............ac.GI.b# | |
###d#e#f################ | |
###A#B#C################ | |
###g#h#i################ | |
########################` // 81 | |
//log(World(world4)) | |
worldN = ` | |
################################################################################# | |
#.....#.....#z#...C.....#.........#.....#.#.....#.......V.#.....#.........#b....# | |
###.#.###.#.#.#.#####.###.#.#######.###.#.#.#.#.#.#######.#####.#.#######.#.#.### | |
#...#..y..#.#.#.#...#.#...#........p#...#...#.#...#.....#.#...T.#...#...#...#...# | |
#.#.#######.#.#.#.###.#.#############.#.#####.#####.#####.#.#.###.###.#.#.#####H# | |
#.#.#.....#.#.#.#...#...#..l....#.....#.#...Y.#...#.......#.#.#...#...#.#.....#.# | |
#.#.#.###.#.#.#.###.#####.###.###.#####.#.#####.###.#######.###.###.###.#######.# | |
#.#.#.G.#.#...#...#.........#...#.#.....#.#...#...#.#.#.........#...#.#...#.....# | |
#.#####.#.#######.#############.#.#####.#.#.###.#.#.#.#.#####.###.###.###.#.###.# | |
#.....#.#.......#.........#.....#.....#.#.#.#...#...#...#...#...#.#.......#...#.# | |
#.#.#.#.#######.#########.#.###.#####.###.#.#.###########.#.###.#.###.#######.#.# | |
#.#.#.#.#.....#.....#...#...#.#.#...#...#.#...............#...#.#...#.........#.# | |
###.###.###.#.###.###.#.#####.#.#.#####.#.###################.#####.###########.# | |
#...#...#...#.#...#...#...#...#...#.....#...#...#...........#.....#.#..g..#...#.# | |
#A###.###.###.#.#.#.###.###.#.###.#.###.###Z#.#.#.#########.#####.#.#.###.#.#.#.# | |
#...#...#.#...#.#.#.#.#.....#.....#.#.#.#.#...#..n#.......#.....#i..#...#...#.#.# | |
#.#.###.#.#.###.###.#.#############.#.#.#.#########.#.#########N#######.#####.#.# | |
#.#...#...#...#.#.......#...........#.#.#.#.X.....#.#.#.......#....o#.#.#...#.#.# | |
#.###.#######.#.#.#######.###########.#.#.#.###.###.#.#.#####.#####.#.#.###.#.#.# | |
#v#.#.#.....#.#...#.....#.......#.#.....#.#...#..u..#.#.#e....O...#.#.#.#...#...# | |
#.#.#.#.###.#.#####.###.#.#####.#.#.#####.###.#######.#.###.#####.#.#.#W#.#.##### | |
#...#.#...#.#.#.#...#...#.....#.#.#.#...#.........#...#...#.#.#.#.#.#.#...#.#...# | |
#####.#.###.#.#.#.#######.#####.#.#.###.#.#########.#####.###.#.#.#.#.#####.#.### | |
#.....#.#...#...#.#.......#.....#.#.....#.....#...#.#.....#...#.....#.....#.#...# | |
#.#####.#.#####.#.###.#####.#####.#####.#####.#.#.#.#.###.#D#########.#####.###.# | |
#.I.#...#.....#.#...#...#...#.......#...#.K.#.#.#...#q..#.#.#......d#.....#...#.# | |
#.#.#.#####.#.#.###.#####.###.###.###.#####.#.#.#######.#E#.#.#####.###.#.###.#.# | |
#.#.#.#...#.#...#...#...#...#...#.....#.#...#.#.......#.#.#...#...#.R.#.#...#.#.# | |
#.#.#.#.#.#######.###.#.###.###.#######.#.###.#####.###.#######.#####.#.#.###.#.# | |
#.#...#.#.........#...#.....#.....#...#.#...#.#...#...#.....#r..#.....#.#.#...#.# | |
#######.###########.#############.#.#.#.###.#.#.#.###.###.#.#.#.#.#####.#.#.###.# | |
#w..#...#.........#.....#.....#...#.#...#...#.#.#.....#.#.#.#.#.#.#...#.#...#...# | |
#.#.#.###.#######.###.#.#.###.#.###.###.#.###.#.#####.#.#.###Q#.#.###.#.#####.#.# | |
#.#.#.....#.....#...#.#.....#.#.....#.#.#...#.#...#.....#.....#...#...#.......#.# | |
#.#.#######.###.###.#.#######.#######.#.#.#.#.###.#####.###########.#.#########.# | |
#.#.#.......#.#...#.#.#.....#...#.....#.#.#.#...#...#.....J.........#...#...#...# | |
#.#.#.#######.#.###.###.###.###.#.###.#.#.#.###.###.###############.#####.#.#.### | |
#.#...#...#...#.....#...#.#.....#.#.#...#.#.....#.#.......#...#...#.#...#.#.#.#.# | |
#.#####.#.#.#.#######.###.#######.#.#####.#######.#######.#.#.#.#.###.#.#.#.#.#.# | |
#.......#...#.........#.................................#...#...#.....#...#.....# | |
#######################################.@.####################################### | |
#.....#.......#.#.........#...#...#...#...........#.........#...................# | |
#.###.#.#####.#.#.#####.#.#.#.#.#.#.#.#.#.#.#######.#######.#.#####.#######.###.# | |
#...#...#...#...#.#...#.#...#...#...#...#.#.#...#...#...#.#...#...#.#.....#.#...# | |
#.#.###.#.#.#####.#.#.#.#####.#########.#.#.#.#.#.###.#.#.#####.#.#.#.#####.#.### | |
#.#...#.#.#.......#.#.#.#.....#.......#.#.#.#.#...#...#...#...#.#.#...#...#.#.#.# | |
#.###.###.###########.#.#.#####.#####.###.#.#.#####.#######.#.#.#.#####.#.#.#.#.# | |
#...#...#.#.......#...#.#.#.#...#...#...#.#...#.....#.......#...#...#...#.#.#.#.# | |
###.###.#.#####.#.#.###.#.#.#.###.#.#.#.#.#####.#####.#############.#.###.#.#.#.# | |
#.#.#.#.........#.#...#.#...#.#...#.#.#.#.#.........#.....#.......#...#.#...#..m# | |
#.#.#.###########.###.#.###.#.#.###.###.#.###.#####.#####.#.#####.#####.###.##### | |
#...#.#.....#...#.#...#.#...#.#.#.#...#.#k..#.#.......#...#.#...#...#.......#...# | |
#.###.#.###.#.###.#.###.#.###.#.#.###.#.###.#.#.#####.#.###.#.#.###.###.#####.#.# | |
#...#j#.#.#.#.#...#.#...#.#...#.#.#...#.#...#.#.#.....#.......#...#...#.#...S.#.# | |
###.#.#.#.#.#.#.###.#.#####.###.#.#.###.#.#####.#.#############.#####.###.#####.# | |
#.#...#.#.#.#.......#.......#...#.#.#...#.......#.#...#...#.....#.....#...#...#.# | |
#.###.#.#.#.#######.#########.###.#.#.#.###.#######.#.#.#.###.###.#.###.#####.#.# | |
#.....#...#...#.M.........#...#.....#.#.#...#.......#...#...#...#.#.#...#.....#.# | |
#.#######.###.#.#####.#####.###.#####.#.#.###.#############.#####.###.###.#####.# | |
#...#.#...#.#.#.#...#.#...#.#.........#.#.#.#.....#.......#....a#...#.#...#...#.# | |
###.#.#.###.#.###.#.###.#.#.###########.#.#.#####.#.#####.#####.#.#.#.#.###.#.#.# | |
#.#...#....t#.....#.#.#.#.#...........#.#...#...#.#.....#...#...#.#...#.#...#...# | |
#.###.#####.#######.#.#.#.#.#########.#.###.###.#.#.###.#########.#####.#.#####.# | |
#.#...#...#.#...#.#...#.#.#...#.....#.#.#.#...#.#.#...#...#.....#...#...#.#...#.# | |
#.#.###.#.#.#.#.#.#####.#.###.#.#####.#.#.###.#.#.###.###.#.#.#####.###.#.#.#.#.# | |
#...#.#.#...#.#.....#...#.#...#.#.....#.#...#.#.#...#...#.#.#.#.........#.#.#.#.# | |
#.###.#.#####.#####.###.#.#.###.#.#####.#.#.#.#.###.###.#.#.#.#.#########.#.###.# | |
#.....#.#.........#.....#.#.....#...#...#.#.#x#...#.#...#...#...#.........#..f#.# | |
#####.#.#####.###.#######.#####.###.#.#####.#.#.#.#.#####.#######.#########.#.#.# | |
#.#...#.#...#.#...#...#...#.....#.#.#...#...#.#.#.#.#...#...L...#.#.......#.#...# | |
#.#.###.#.#.###.###.#.#.###.#####.#.#####.###.###.#.#B#.#########.###.###.#.##### | |
#.#.P.#...#.....#.#.#.#...#.......#.....#...#.....#...#.........#...#...#.....#.# | |
#.###.###########.#.#.###.#############.###.#####.#############.###.#.#######.#.# | |
#c....#...#.#.......#...#.#.......#...#.#.....#...#...#.....#.#.#...#.#...#.....# | |
#.#####.#.#.#.#######.###.#.#####.#.#.#.#.#.###.###.#.#.#.#.#.#.#.#####.#.#####.# | |
#...#..s#.#.F.....#...#...#.#...#.#.#...#.#.#...#...#...#.#.#.#.#.......#.....#.# | |
###.###.#.#######.#.#.#.###.#.###.#.###.#.###.###########.#.#.#.#####.#######.### | |
#...#...#.#...#...#.#.#...#.#...#...#.#.#.....#.......#...#...#.#...#.#.....#...# | |
#.###.###.#.#.#####.#.###.#.#.#.#####.#.#.#######.###.#.#####.#.#.#.###.###.###.# | |
#.....#.....#.......#...#...#.#.........#.....U...#.....#.....#...#....h#.......# | |
#################################################################################` //4700 | |
log(World(worldN)) |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment