@@ -3,7 +3,6 @@ package main
3
3
import (
4
4
"fmt"
5
5
"os"
6
- "slices"
7
6
"strings"
8
7
)
9
8
@@ -21,195 +20,191 @@ var nextDirs = map[string]string{
21
20
"<" : "^" ,
22
21
}
23
22
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 )
33
31
}
32
+ return c
33
+ }
34
34
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
+ }
43
43
}
44
-
45
- return r , c , grid , visited
44
+ return out
46
45
}
47
46
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
49
51
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
57
58
}
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
78
65
}
79
-
66
+ // step in current direction
80
67
grid [r ][c ] = "."
81
- r = nextR
82
- c = nextC
83
- grid [r ][c ] = nextDir
68
+ grid [nextR ][nextC ] = curDir
84
69
85
- return r , c , grid , visitedDirs , couldLoop
70
+ return nextR , nextC , false
86
71
}
87
72
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
110
84
}
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
118
90
}
119
- return false
120
91
}
121
- return false
122
92
}
123
93
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
-
134
94
func part1 () {
135
95
raw , _ := os .ReadFile ("input.txt" )
136
96
//raw, _ := os.ReadFile("test.txt")
137
97
data := string (raw )
138
98
139
- grid := [][]string {}
140
- visited := [][]string {}
99
+ origGrid := [][]string {}
141
100
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 , "" ))
144
102
}
145
103
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
153
112
break
154
113
}
155
114
}
156
115
}
157
116
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
+ }
167
134
}
135
+ fmt .Println (count )
136
+
137
+ return
168
138
}
169
139
}
170
- fmt .Println (tot )
171
140
}
172
141
173
142
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")
176
145
data := string (raw )
177
146
178
- grid := [][]string {}
179
- visitedDirs := [][][]string {}
147
+ origGrid := [][]string {}
180
148
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 , "" ))
187
150
}
188
151
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
196
160
break
197
161
}
198
162
}
199
163
}
200
164
165
+ visited := emptyVisited (len (origGrid ))
166
+ grid := copyGrid (origGrid )
201
167
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 {
206
202
count += 1
207
203
}
208
204
}
209
- fmt .Println (count )
210
205
}
211
206
212
207
func main () {
213
- // part1()
208
+ part1 ()
214
209
part2 ()
215
210
}
0 commit comments