Skip to content

Commit befced2

Browse files
committedDec 9, 2024
finally getting around to cleaning up day 6
1 parent 3943644 commit befced2

File tree

2 files changed

+133
-319
lines changed

2 files changed

+133
-319
lines changed
 

‎6/main.go

+133-138
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@ package main
33
import (
44
"fmt"
55
"os"
6-
"slices"
76
"strings"
87
)
98

@@ -21,195 +20,191 @@ var nextDirs = map[string]string{
2120
"<": "^",
2221
}
2322

24-
func move(r, c int, grid, visited [][]string) (int, int, [][]string, [][]string) {
25-
visited[r][c] = "X"
26-
curDir := grid[r][c]
27-
dirCoord := dirCoords[curDir]
28-
nextR := r + dirCoord[0]
29-
nextC := c + dirCoord[1]
30-
31-
if !(nextR >= 0 && nextR < len(grid) && nextC >= 0 && nextC < len(grid[0])) {
32-
return nextR, nextC, grid, visited
23+
func copyGrid(grid [][]string) [][]string {
24+
c := [][]string{}
25+
for i := range len(grid) {
26+
row := []string{}
27+
for j := range len(grid[0]) {
28+
row = append(row, grid[i][j])
29+
}
30+
c = append(c, row)
3331
}
32+
return c
33+
}
3434

35-
if grid[nextR][nextC] == "#" { // need to turn
36-
nextDir := nextDirs[curDir]
37-
grid[r][c] = nextDir
38-
} else { // stay same direction
39-
grid[r][c] = "."
40-
r = nextR
41-
c = nextC
42-
grid[r][c] = curDir
35+
// visited contains which direction we moved in at each coordinate
36+
func emptyVisited(n int) [][]map[string]bool {
37+
out := make([][]map[string]bool, n)
38+
for i := range n {
39+
out[i] = make([]map[string]bool, n)
40+
for j := range n {
41+
out[i][j] = make(map[string]bool)
42+
}
4343
}
44-
45-
return r, c, grid, visited
44+
return out
4645
}
4746

48-
func move2(r, c int, grid [][]string, visitedDirs [][][]string) (int, int, [][]string, [][][]string, bool) {
47+
// updates grid in place by moving the arrow one step, updates visited in place with position at start of step
48+
// returns r, c, bool where bool indicates out of bounds if were to take a step, ie you are on the very edge
49+
func step(r, c int, gridPtr *[][]string, visited *[][]map[string]bool) (int, int, bool) {
50+
grid := *gridPtr
4951
curDir := grid[r][c]
50-
dirCoord := dirCoords[curDir]
51-
visitedDirs[r][c] = append(visitedDirs[r][c], curDir)
52-
nextR := r + dirCoord[0]
53-
nextC := c + dirCoord[1]
54-
55-
if !(nextR >= 0 && nextR < len(grid) && nextC >= 0 && nextC < len(grid[0])) {
56-
return nextR, nextC, grid, visitedDirs, false
52+
// make location as visited with the current direction then take a step if valid
53+
(*visited)[r][c][curDir] = true
54+
nextR, nextC := r+dirCoords[curDir][0], c+dirCoords[curDir][1]
55+
nextInBounds := nextR >= 0 && nextR < len(grid) && nextC >= 0 && nextC < len(grid[0])
56+
if !nextInBounds {
57+
return r, c, true
5758
}
58-
59-
couldLoop := false
60-
nextDir := ""
61-
if grid[nextR][nextC] == "#" { // need to turn
62-
nextDir = nextDirs[curDir]
63-
nextDirCoord := dirCoords[nextDir]
64-
nextR = r + nextDirCoord[0]
65-
nextC = c + nextDirCoord[1]
66-
67-
grid[r][c] = nextDir
68-
} else { // stay same direction
69-
nextDir = curDir
70-
// check if a loop is possible by imagining we need to turn and see if we follow that new direction we see a visitedDir
71-
// that matches the new direction
72-
loopDir := nextDirs[curDir]
73-
loopDirCoord := dirCoords[loopDir]
74-
loopR := r + loopDirCoord[0]
75-
loopC := c + loopDirCoord[1]
76-
77-
couldLoop = checkLoop(loopR, loopC, loopDir, grid, visitedDirs)
59+
// check for turn
60+
if grid[nextR][nextC] == "#" {
61+
// stay in place and turn
62+
turnedDir := nextDirs[curDir]
63+
grid[r][c] = turnedDir
64+
return r, c, false
7865
}
79-
66+
// step in current direction
8067
grid[r][c] = "."
81-
r = nextR
82-
c = nextC
83-
grid[r][c] = nextDir
68+
grid[nextR][nextC] = curDir
8469

85-
return r, c, grid, visitedDirs, couldLoop
70+
return nextR, nextC, false
8671
}
8772

88-
func checkLoop(r, c int, loopDir string, grid [][]string, visitedDirs [][][]string) bool {
89-
if loopDir == "^" {
90-
for i := r; i >= 0; i-- {
91-
if slices.Contains(visitedDirs[i][c], loopDir) {
92-
return true
93-
}
94-
}
95-
return false
96-
}
97-
if loopDir == "v" {
98-
for i := r; i < len(visitedDirs); i++ {
99-
if slices.Contains(visitedDirs[i][c], loopDir) {
100-
return true
101-
}
102-
}
103-
return false
104-
}
105-
if loopDir == ">" {
106-
for i := c; i < len(visitedDirs[0]); i++ {
107-
if slices.Contains(visitedDirs[r][i], loopDir) {
108-
return true
109-
}
73+
// takes in a grid with a start location. traverses the grid till it exits or gets in a loop.
74+
// returns true if has a loop
75+
func traverse(r, c int, origGrid [][]string) bool {
76+
grid := copyGrid(origGrid)
77+
visited := emptyVisited(len(origGrid))
78+
// break if make it out of grid or get in a loop
79+
for {
80+
done := false
81+
r, c, done = step(r, c, &grid, &visited)
82+
if done {
83+
return false
11084
}
111-
return false
112-
}
113-
if loopDir == "<" {
114-
for i := c; i >= 0; i-- {
115-
if slices.Contains(visitedDirs[r][i], loopDir) {
116-
return true
117-
}
85+
86+
curDir := grid[r][c]
87+
looped := visited[r][c][curDir]
88+
if looped {
89+
return true
11890
}
119-
return false
12091
}
121-
return false
12292
}
12393

124-
//func checkLoop(r, c int, grid [][]string, visitedDirs [][][]string) bool {
125-
// for r >= 0 && r < len(grid) && c >= 0 && c < len(grid[0]) {
126-
// r, c, grid, visitedDirs, hasLoop = move2(r, c, grid, visitedDirs)
127-
// if hasLoop {
128-
// return true
129-
// }
130-
// }
131-
// return false
132-
//}
133-
13494
func part1() {
13595
raw, _ := os.ReadFile("input.txt")
13696
//raw, _ := os.ReadFile("test.txt")
13797
data := string(raw)
13898

139-
grid := [][]string{}
140-
visited := [][]string{}
99+
origGrid := [][]string{}
141100
for _, row := range strings.Split(data, "\n") {
142-
grid = append(grid, strings.Split(row, ""))
143-
visited = append(visited, strings.Split(row, ""))
101+
origGrid = append(origGrid, strings.Split(row, ""))
144102
}
145103

146-
r := -1
147-
c := -1
148-
for i := range len(grid) {
149-
for j := range len(grid[0]) {
150-
if grid[i][j] == "^" {
151-
r = i
152-
c = j
104+
// find start
105+
startR := -1
106+
startC := -1
107+
for i := range len(origGrid) {
108+
for j := range len(origGrid[0]) {
109+
if origGrid[i][j] == "^" {
110+
startR = i
111+
startC = j
153112
break
154113
}
155114
}
156115
}
157116

158-
for r >= 0 && r < len(grid) && c >= 0 && c < len(grid[0]) {
159-
r, c, grid, visited = move(r, c, grid, visited)
160-
}
161-
162-
tot := 0
163-
for i := range len(visited) {
164-
for j := range len(visited[0]) {
165-
if visited[i][j] == "X" {
166-
tot += 1
117+
visited := emptyVisited(len(origGrid))
118+
grid := copyGrid(origGrid)
119+
r := startR
120+
c := startC
121+
for {
122+
// walk through the guards path and at each step imagine there is an obstacle in front of us and see if it would cause
123+
// a loop
124+
done := false
125+
r, c, done = step(r, c, &grid, &visited)
126+
if done {
127+
count := 0
128+
for i := range len(visited) {
129+
for j := range len(visited[0]) {
130+
if len(visited[i][j]) > 0 {
131+
count++
132+
}
133+
}
167134
}
135+
fmt.Println(count)
136+
137+
return
168138
}
169139
}
170-
fmt.Println(tot)
171140
}
172141

173142
func part2() {
174-
//raw, _ := os.ReadFile("input.txt")
175-
raw, _ := os.ReadFile("test.txt")
143+
raw, _ := os.ReadFile("input.txt")
144+
//raw, _ := os.ReadFile("test.txt")
176145
data := string(raw)
177146

178-
grid := [][]string{}
179-
visitedDirs := [][][]string{}
147+
origGrid := [][]string{}
180148
for _, row := range strings.Split(data, "\n") {
181-
grid = append(grid, strings.Split(row, ""))
182-
dirRow := [][]string{}
183-
for range len(strings.Split(row, "")) {
184-
dirRow = append(dirRow, []string{})
185-
}
186-
visitedDirs = append(visitedDirs, dirRow)
149+
origGrid = append(origGrid, strings.Split(row, ""))
187150
}
188151

189-
r := -1
190-
c := -1
191-
for i := range len(grid) {
192-
for j := range len(grid[0]) {
193-
if grid[i][j] == "^" {
194-
r = i
195-
c = j
152+
// find start
153+
startR := -1
154+
startC := -1
155+
for i := range len(origGrid) {
156+
for j := range len(origGrid[0]) {
157+
if origGrid[i][j] == "^" {
158+
startR = i
159+
startC = j
196160
break
197161
}
198162
}
199163
}
200164

165+
visited := emptyVisited(len(origGrid))
166+
grid := copyGrid(origGrid)
201167
count := 0
202-
couldLoop := false
203-
for r >= 0 && r < len(grid) && c >= 0 && c < len(grid[0]) {
204-
r, c, grid, visitedDirs, couldLoop = move2(r, c, grid, visitedDirs)
205-
if couldLoop {
168+
r := startR
169+
c := startC
170+
for {
171+
// walk through the guards path and at each step imagine there is an obstacle in front of us and see if it would cause
172+
// a loop
173+
done := false
174+
r, c, done = step(r, c, &grid, &visited)
175+
if done {
176+
fmt.Println(count)
177+
return
178+
}
179+
180+
curDir := grid[r][c]
181+
withObstacle := copyGrid(origGrid)
182+
// move start to current position
183+
withObstacle[startR][startC] = "."
184+
withObstacle[r][c] = curDir
185+
186+
// make sure obstacle goes in a valid spot
187+
dirCoord := dirCoords[curDir]
188+
obstacleR, obstacleC := r+dirCoord[0], c+dirCoord[1]
189+
nextInBounds := obstacleR >= 0 && obstacleR < len(origGrid) && obstacleC >= 0 && obstacleC < len(origGrid[0])
190+
if !nextInBounds {
191+
fmt.Println(count)
192+
return
193+
}
194+
// this means we already visited where the obstacle would go so we would have ran into ito earlier so skip it
195+
if len(visited[obstacleR][obstacleC]) > 0 {
196+
continue
197+
}
198+
// place obstacle, and check if we started from there would we get in a loop
199+
withObstacle[obstacleR][obstacleC] = "#"
200+
hasLoop := traverse(r, c, withObstacle)
201+
if hasLoop {
206202
count += 1
207203
}
208204
}
209-
fmt.Println(count)
210205
}
211206

212207
func main() {
213-
//part1()
208+
part1()
214209
part2()
215210
}

0 commit comments

Comments
 (0)
Please sign in to comment.