parent
2ac1c6069a
commit
51c91ab2f2
@ -0,0 +1,61 @@
|
|||||||
|
def succ(coord, delta)
|
||||||
|
coord.zip(delta).map { _1 + _2 }
|
||||||
|
end
|
||||||
|
|
||||||
|
map = ARGF.readlines(chomp: true)
|
||||||
|
.flat_map.with_index {|row, y|
|
||||||
|
row.chars.map.with_index {|elem, x|
|
||||||
|
[[y,x], elem.to_i]
|
||||||
|
}
|
||||||
|
}.to_h
|
||||||
|
|
||||||
|
max_y = map.keys.map(&:first).max
|
||||||
|
max_x = map.keys.map(&:last).max
|
||||||
|
dest = [max_y,max_x]
|
||||||
|
|
||||||
|
Node = Data.define(:pos, :dir, :continued)
|
||||||
|
|
||||||
|
frontier = []
|
||||||
|
visited = {}
|
||||||
|
|
||||||
|
# seed initial directions
|
||||||
|
[
|
||||||
|
[1,0], # down
|
||||||
|
[0,1], # right
|
||||||
|
].each do |dir|
|
||||||
|
node = Node.new(dir, dir, 1)
|
||||||
|
frontier << node
|
||||||
|
visited[node] = map.fetch(dir)
|
||||||
|
end
|
||||||
|
|
||||||
|
loop do
|
||||||
|
frontier.sort_by! { visited.fetch(_1) }
|
||||||
|
current = frontier.shift
|
||||||
|
|
||||||
|
# turn left and right
|
||||||
|
nexts = [current.dir.reverse, current.dir.reverse.map { -_1 }].map {|turn|
|
||||||
|
Node.new(current.pos.zip(turn).map { _1 + _2 }, turn, 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
# keep going?
|
||||||
|
if current.continued < 3
|
||||||
|
nexts << Node.new(
|
||||||
|
current.pos.zip(current.dir).map { _1 + _2 },
|
||||||
|
current.dir,
|
||||||
|
current.continued + 1,
|
||||||
|
)
|
||||||
|
end
|
||||||
|
|
||||||
|
if found = nexts.find { _1.pos == dest }
|
||||||
|
p visited.fetch(current) + map.fetch(dest)
|
||||||
|
exit
|
||||||
|
end
|
||||||
|
|
||||||
|
nexts
|
||||||
|
.select { map.has_key?(_1.pos) }
|
||||||
|
.reject { visited.has_key?(_1) }
|
||||||
|
.each do |node|
|
||||||
|
frontier << node
|
||||||
|
visited[node] = visited.fetch(current) + map.fetch(node.pos)
|
||||||
|
end
|
||||||
|
end
|
Loading…
Reference in new issue