-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathmain.go
93 lines (80 loc) · 2.07 KB
/
main.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
package main
import (
"io"
"github.com/teivah/go-aoc"
)
type cell struct {
empty bool
frequency rune
}
func fs1(input io.Reader) int {
antennas := make(map[rune][]aoc.Position)
board := aoc.ParseBoard(aoc.ReaderToStrings(input), func(r rune, pos aoc.Position) cell {
switch r {
case '.':
return cell{empty: true}
default:
antennas[r] = append(antennas[r], pos)
return cell{frequency: r}
}
})
antinodes := make(map[aoc.Position]struct{})
for _, positions := range antennas {
findAntinodes(board, positions, antinodes, setAntinode)
}
return len(antinodes)
}
func findAntinodes(board aoc.Board[cell], positions []aoc.Position, antinodes map[aoc.Position]struct{}, f func(board aoc.Board[cell], x, y aoc.Position, antinodes map[aoc.Position]struct{})) {
for i := 0; i < len(positions); i++ {
for j := i + 1; j < len(positions); j++ {
x := positions[i]
y := positions[j]
f(board, x, y, antinodes)
f(board, y, x, antinodes)
}
}
}
func setAntinode(board aoc.Board[cell], x, y aoc.Position, antinodes map[aoc.Position]struct{}) {
d := delta(x, y)
z := delta(y, d)
if board.Contains(z) {
antinodes[z] = struct{}{}
}
}
func delta(x, y aoc.Position) aoc.Position {
return aoc.Position{
Row: x.Row - y.Row,
Col: x.Col - y.Col,
}
}
func fs2(input io.Reader) int {
antennas := make(map[rune][]aoc.Position)
board := aoc.ParseBoard(aoc.ReaderToStrings(input), func(r rune, pos aoc.Position) cell {
switch r {
case '.':
return cell{empty: true}
default:
antennas[r] = append(antennas[r], pos)
return cell{frequency: r}
}
})
antinodes := make(map[aoc.Position]struct{})
for _, positions := range antennas {
if len(positions) <= 1 {
continue
}
findAntinodes(board, positions, antinodes, setAntinode2)
for _, pos := range positions {
antinodes[pos] = struct{}{}
}
}
return len(antinodes)
}
func setAntinode2(board aoc.Board[cell], x, y aoc.Position, antinodes map[aoc.Position]struct{}) {
d := delta(x, y)
z := delta(y, d)
if board.Contains(z) {
antinodes[z] = struct{}{}
setAntinode2(board, y, z, antinodes)
}
}