advent-of-code-2023/day9/main.go

114 lines
2.3 KiB
Go
Raw Normal View History

2023-12-09 18:30:52 +00:00
package main
import (
"fmt"
"io"
"os"
2023-12-09 18:33:54 +00:00
"slices"
2023-12-09 18:30:52 +00:00
"strconv"
"strings"
)
func main() {
if len(os.Args) != 2 && len(os.Args) != 3 {
fmt.Fprintf(os.Stderr, "Usage: %s inputfile\n", os.Args[0])
os.Exit(1)
}
inputFilename := os.Args[1]
inputFile, err := os.Open(inputFilename)
if err != nil {
panic(fmt.Sprintf("could not open input file: %s", err))
}
defer inputFile.Close()
inputBytes, err := io.ReadAll(inputFile)
if err != nil {
panic(fmt.Sprintf("could not read input file: %s", err))
}
input := strings.TrimSpace(string(inputBytes))
inputLines := strings.Split(input, "\n")
histories, err := parseHistories(inputLines)
if err != nil {
panic(fmt.Sprintf("failed to parse histories: %s", err))
}
fmt.Printf("Part 1: %d\n", part1(histories))
2023-12-09 18:33:54 +00:00
fmt.Printf("Part 2: %d\n", part2(histories))
2023-12-09 18:30:52 +00:00
}
func part1(histories [][]int) int {
total := 0
for _, history := range histories {
total += predictNextValue(history)
}
return total
}
2023-12-09 18:33:54 +00:00
func part2(histories [][]int) int {
total := 0
for _, history := range histories {
reversedHistory := slices.Clone(history)
slices.Reverse(reversedHistory)
total += predictNextValue(reversedHistory)
}
return total
}
2023-12-09 18:30:52 +00:00
// predictNextValue calculates the next item in the sequence by using the nth differences
func predictNextValue(history []int) int {
if allEqual(history, 0) {
return 0
}
differences := make([]int, len(history)-1)
for i := range history[1:] {
differences[i] = history[i+1] - history[i]
}
return history[len(history)-1] + predictNextValue(differences)
}
func allEqual[T comparable, S ~[]T](values S, shouldEqual T) bool {
for _, value := range values {
if value != shouldEqual {
return false
}
}
return true
}
func parseHistories(lines []string) ([][]int, error) {
histories := make([][]int, len(lines))
for i, line := range lines {
history, err := parseHistory(line)
if err != nil {
return nil, fmt.Errorf("parse history %d: %s", i, err)
}
histories[i] = history
}
return histories, nil
}
func parseHistory(line string) ([]int, error) {
rawNums := strings.Split(line, " ")
history := make([]int, len(rawNums))
for i, rawNum := range rawNums {
n, err := strconv.Atoi(rawNum)
if err != nil {
return nil, fmt.Errorf("parse item %d: %w", i, err)
}
history[i] = n
}
return history, nil
}