Skip to content

Commit f4db3e4

Browse files
day 12
1 parent 109af36 commit f4db3e4

File tree

3 files changed

+208
-0
lines changed

3 files changed

+208
-0
lines changed

day12/hill-climping-algorithm.ts

+107
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,107 @@
1+
import * as fs from "fs";
2+
3+
/**
4+
* If true prints the map with the shortest route from start to finish marked.
5+
*/
6+
const PRINT_ROUTE = true;
7+
8+
let endPoint: MapPoint | undefined;
9+
class MapPoint {
10+
public isStart = false;
11+
public height: number;
12+
public visited = false;
13+
public distanceFromDestination = 0;
14+
// Variables needed only for printing the map.
15+
public parent?: MapPoint;
16+
public onShortestRoute = false;
17+
18+
constructor(public x: number, public y: number, char: string) {
19+
if (char === "S") {
20+
this.isStart = true;
21+
char = "a";
22+
} else if (char === "E") {
23+
// Starting from the end.
24+
this.visited = true;
25+
// eslint-disable-next-line @typescript-eslint/no-this-alias
26+
endPoint = this;
27+
char = "z";
28+
}
29+
this.height = char.charCodeAt(0);
30+
}
31+
32+
/**
33+
* Checks if neighbor is achievable from this map point and sets its parameters.
34+
* @param neighbor mapPoint to which we want to travel from this
35+
* @param queue - neighbor is added to the queue if we can travel to him
36+
* @returns true if found destination, false otherwise.
37+
*/
38+
public processNeighbor(neighbor: MapPoint, queue: MapPoint[]): boolean {
39+
if (neighbor && !neighbor.visited && this.height - neighbor.height <= 1) {
40+
if (neighbor.isStart) {
41+
// Parent is required only for the route printing.
42+
let parent = this.parent;
43+
while (parent) {
44+
parent.onShortestRoute = true;
45+
parent = parent.parent;
46+
}
47+
return true;
48+
}
49+
neighbor.visited = true;
50+
neighbor.distanceFromDestination = this.distanceFromDestination + 1;
51+
queue.push(neighbor);
52+
neighbor.parent = this;
53+
}
54+
return false;
55+
}
56+
}
57+
58+
const data = fs.readFileSync("./input.txt", "utf-8");
59+
const heightMap = data
60+
.split(/\r?\n/)
61+
.map((row, x) =>
62+
row.split("").map((height, y) => new MapPoint(x, y, height)),
63+
);
64+
65+
/**
66+
* Solution is the BFS from the hill top.
67+
*/
68+
let shortestPathB = -1;
69+
function shortestPath() {
70+
if (endPoint === undefined) {
71+
throw new Error("There is no start point on the map!");
72+
}
73+
const queue = [endPoint];
74+
for (const point of queue) {
75+
if (shortestPathB < 0 && point.height === "a".charCodeAt(0))
76+
shortestPathB = point.distanceFromDestination;
77+
if (
78+
point.processNeighbor(heightMap[point.x]?.[point.y + 1], queue) ||
79+
point.processNeighbor(heightMap[point.x]?.[point.y - 1], queue) ||
80+
point.processNeighbor(heightMap[point.x + 1]?.[point.y], queue) ||
81+
point.processNeighbor(heightMap[point.x - 1]?.[point.y], queue)
82+
)
83+
return point.distanceFromDestination + 1;
84+
}
85+
}
86+
87+
const shortestPathA = shortestPath();
88+
// Is the starting position is also a first point with elevation "a" from the top, shortestPathB won't be set.
89+
if (shortestPathA && shortestPathA < shortestPathB)
90+
shortestPathB = shortestPathA;
91+
92+
if (PRINT_ROUTE) {
93+
// Printing the map with the shortest path marked with red.
94+
let map = "\n";
95+
for (const row of heightMap) {
96+
for (const point of row) {
97+
if (point.onShortestRoute)
98+
map += `\x1b[31m${String.fromCharCode(point.height)}\x1b[0m`;
99+
else map += String.fromCharCode(point.height);
100+
}
101+
map += "\n";
102+
}
103+
console.log(map);
104+
}
105+
106+
console.log(`Part 1: ${shortestPathA}`);
107+
console.log(`Part 2: ${shortestPathB}`);

day12/input.txt

+41
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
abaacccccccccccccaaaaaaaccccccccccccccccccccccccccccccccccaaaaaa
2+
abaaccccccccccccccaaaaaaaaaaccccccccccccccccccccccccccccccccaaaa
3+
abaaaaacccccccccaaaaaaaaaaaaccccccccccccccccccccccccccccccccaaaa
4+
abaaaaaccccccccaaaaaaaaaaaaaacccccccccccccccccdcccccccccccccaaaa
5+
abaaaccccccccccaaaaaaaaccacacccccccccccccccccdddcccccccccccaaaaa
6+
abaaacccccccccaaaaaaaaaaccaaccccccccccccciiiiddddcccccccccccaccc
7+
abcaaaccccccccaaaaaaaaaaaaaaccccccccccciiiiiijddddcccccccccccccc
8+
abccaaccccccccaccaaaaaaaaaaaacccccccccciiiiiijjddddccccaaccccccc
9+
abccccccccccccccaaacaaaaaaaaaaccccccciiiiippijjjddddccaaaccccccc
10+
abccccccccccccccaacccccaaaaaaacccccciiiippppppjjjdddddaaaaaacccc
11+
abccccccccccccccccccccaaaaaaccccccckiiippppppqqjjjdddeeeaaaacccc
12+
abccccccccccccccccccccaaaaaaccccckkkiippppuupqqjjjjdeeeeeaaccccc
13+
abccccccccccccccccccccccccaaccckkkkkkipppuuuuqqqjjjjjeeeeeaccccc
14+
abccccccccccccccccccccccccccckkkkkkoppppuuuuuvqqqjjjjjkeeeeccccc
15+
abcccccccccccccccccccccccccckkkkooooppppuuxuvvqqqqqqjkkkeeeecccc
16+
abccaaccaccccccccccccccccccckkkoooooopuuuuxyvvvqqqqqqkkkkeeecccc
17+
abccaaaaacccccaaccccccccccckkkoooouuuuuuuxxyyvvvvqqqqqkkkkeecccc
18+
abcaaaaacccccaaaacccccccccckkkooouuuuxxxuxxyyvvvvvvvqqqkkkeeeccc
19+
abcaaaaaaaaaaaaacccccccccccjjjooottuxxxxxxxyyyyyvvvvrrrkkkeecccc
20+
abcccaaaacaaaaaaaaacaaccccccjjoootttxxxxxxxyyyyyyvvvrrkkkfffcccc
21+
SbccaacccccaaaaaaaaaaaccccccjjjooottxxxxEzzzyyyyvvvrrrkkkfffcccc
22+
abcccccccccaaaaaaaaaaaccccccjjjooootttxxxyyyyyvvvvrrrkkkfffccccc
23+
abcaacccccaaaaaaaaaaaccccccccjjjooottttxxyyyyywwvrrrrkkkfffccccc
24+
abaaacccccaaaaaaaaaaaaaacccccjjjjonnttxxyyyyyywwwrrlllkfffcccccc
25+
abaaaaaaaaaaacaaaaaaaaaaccccccjjjnnnttxxyywwyyywwrrlllffffcccccc
26+
abaaaaaaaaaaaaaaaaaaaaaaccccccjjjnntttxxwwwwwywwwrrlllfffccccccc
27+
abaaccaaaaaaaaaaaaaaacccccccccjjjnntttxwwwsswwwwwrrlllfffccccccc
28+
abaacccaaaaaaaacccaaacccccccccjjinnttttwwsssswwwsrrlllgffacccccc
29+
abccccaaaaaaccccccaaaccccccccciiinnntttsssssssssssrlllggaacccccc
30+
abccccaaaaaaaccccccccccaaccccciiinnntttsssmmssssssrlllggaacccccc
31+
abccccaacaaaacccccccaacaaaccccciinnnnnnmmmmmmmsssslllgggaaaacccc
32+
abccccccccaaacccccccaaaaacccccciiinnnnnmmmmmmmmmmllllgggaaaacccc
33+
abaaaccccccccccccccccaaaaaacccciiiinnnmmmhhhmmmmmlllgggaaaaccccc
34+
abaaaaacccccccccccaaaaaaaaaccccciiiiiiihhhhhhhhmmlgggggaaacccccc
35+
abaaaaaccccaaccccaaaaaaacaacccccciiiiihhhhhhhhhhggggggcaaacccccc
36+
abaaaaccccaaaccccaaaacaaaaacccccccciiihhaaaaahhhhggggccccccccccc
37+
abaaaaaaacaaacccccaaaaaaaaaccccccccccccccaaaacccccccccccccccccaa
38+
abaacaaaaaaaaaaaccaaaaaaaaccccccccccccccccaaaccccccccccccccccaaa
39+
abcccccaaaaaaaaacccaaaaaaaccccccccccccccccaacccccccccccccccccaaa
40+
abccccccaaaaaaaaaaaaaaaaacccccccccccccccccaaacccccccccccccaaaaaa
41+
abcccccaaaaaaaaaaaaaaaaaaaaaccccccccccccccccccccccccccccccaaaaaa

day12/problem.txt

+60
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,60 @@
1+
--- Day 12: Hill Climbing Algorithm ---
2+
3+
You try contacting the Elves using your handheld device, but the river you're following must be too low to get a decent signal.
4+
5+
You ask the device for a heightmap of the surrounding area (your puzzle input). The heightmap shows the local area from above broken into a grid; the elevation of each square of the grid is given by a single lowercase letter, where a is the lowest elevation, b is the next-lowest, and so on up to the highest elevation, z.
6+
7+
Also included on the heightmap are marks for your current position (S) and the location that should get the best signal (E). Your current position (S) has elevation a, and the location that should get the best signal (E) has elevation z.
8+
9+
You'd like to reach E, but to save energy, you should do it in as few steps as possible. During each step, you can move exactly one square up, down, left, or right. To avoid needing to get out your climbing gear, the elevation of the destination square can be at most one higher than the elevation of your current square; that is, if your current elevation is m, you could step to elevation n, but not to elevation o. (This also means that the elevation of the destination square can be much lower than the elevation of your current square.)
10+
11+
For example:
12+
13+
Sabqponm
14+
abcryxxl
15+
accszExk
16+
acctuvwj
17+
abdefghi
18+
19+
Here, you start in the top-left corner; your goal is near the middle. You could start by moving down or right, but eventually you'll need to head toward the e at the bottom. From there, you can spiral around to the goal:
20+
21+
v..v<<<<
22+
>v.vv<<^
23+
.>vv>E^^
24+
..v>>>^^
25+
..>>>>>^
26+
27+
In the above diagram, the symbols indicate whether the path exits each square moving up (^), down (v), left (<), or right (>). The location that should get the best signal is still E, and . marks unvisited squares.
28+
29+
This path reaches the goal in 31 steps, the fewest possible.
30+
31+
What is the fewest steps required to move from your current position to the location that should get the best signal?
32+
33+
Your puzzle answer was 370.
34+
--- Part Two ---
35+
36+
As you walk up the hill, you suspect that the Elves will want to turn this into a hiking trail. The beginning isn't very scenic, though; perhaps you can find a better starting point.
37+
38+
To maximize exercise while hiking, the trail should start as low as possible: elevation a. The goal is still the square marked E. However, the trail should still be direct, taking the fewest steps to reach its goal. So, you'll need to find the shortest path from any square at elevation a to the square marked E.
39+
40+
Again consider the example from above:
41+
42+
Sabqponm
43+
abcryxxl
44+
accszExk
45+
acctuvwj
46+
abdefghi
47+
48+
Now, there are six choices for starting position (five marked a, plus the square marked S that counts as being at elevation a). If you start at the bottom-left square, you can reach the goal most quickly:
49+
50+
...v<<<<
51+
...vv<<^
52+
...v>E^^
53+
.>v>>>^^
54+
>^>>>>>^
55+
56+
This path reaches the goal in only 29 steps, the fewest possible.
57+
58+
What is the fewest steps required to move starting from any square with elevation a to the location that should get the best signal?
59+
60+
Your puzzle answer was 363.

0 commit comments

Comments
 (0)