123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139 |
- package main
- import (
- "bufio"
- "fmt"
- "log"
- "os"
- )
- type Button struct {
- id byte
- x, y int64
- }
- type Machine struct {
- buttons []Button
- x, y int64
- }
- func readInput(file *os.File) []Machine {
- scanner := bufio.NewScanner(file)
- var machines []Machine
- var machine Machine
- var buttonsRead int
- for scanner.Scan() {
- line := scanner.Text()
- if line == "" {
- buttonsRead = 0
- machines = append(machines, machine)
- machine = Machine{}
- continue
- }
- if buttonsRead < 2 {
- var button Button
- n, err := fmt.Sscanf(line, "Button %c: X+%d, Y+%d", &button.id, &button.x, &button.y)
- if n != 3 || err != nil {
- log.Fatalf("Not able to parse button '%s': %s", line, err)
- }
- machine.buttons = append(machine.buttons, button)
- buttonsRead++
- } else {
- n, err := fmt.Sscanf(line, "Prize: X=%d, Y=%d", &machine.x, &machine.y)
- if n != 2 || err != nil {
- log.Fatalf("Not able to parse machine '%s': %s", line, err)
- }
- }
- }
- machines = append(machines, machine)
- return machines
- }
- func min(a, b int64) int64 {
- if a < b {
- return a
- }
- return b
- }
- func calculate(machine Machine, button int, limit int64, minimum int64) [2]int64 {
- var results [2]int64
- otherButton := (button + 1) % 2
- if machine.x%machine.buttons[button].x == 0 && machine.y%machine.buttons[button].y == 0 {
- pushes := machine.x / machine.buttons[button].x
- if pushes*machine.buttons[button].y == machine.y {
- results[button] = pushes
- results[otherButton] = 0
- return results
- }
- }
- start := min(machine.x/machine.buttons[button].x, machine.y/machine.buttons[button].y)
- if limit > 0 && start > limit {
- start = limit
- }
- for ; start > minimum; start-- {
- deltaX := machine.x - start*machine.buttons[button].x
- if deltaX%machine.buttons[otherButton].x == 0 {
- otherPushes := deltaX / machine.buttons[otherButton].x
- if limit > 0 && otherPushes > limit {
- continue
- }
- if machine.y-start*machine.buttons[button].y != otherPushes*machine.buttons[otherButton].y {
- continue
- }
- results[button] = start
- results[otherButton] = otherPushes
- return results
- }
- }
- return results
- }
- func checkMachine(machine Machine, limit int64, modifier int64, minimum int64) int64 {
- machine.x += modifier
- machine.y += modifier
- resultA := calculate(machine, 0, limit, minimum)
- resultB := calculate(machine, 1, limit, minimum)
- costA := resultA[0]*3 + resultA[1]
- costB := resultB[0]*3 + resultB[1]
- return min(costA, costB)
- }
- func solve(machines []Machine, limit int64, modifier int64, minimum int64) int64 {
- var result int64
- for _, machine := range machines {
- result += checkMachine(machine, limit, modifier, minimum)
- }
- return result
- }
- func main() {
- if len(os.Args) < 2 {
- log.Fatal("You need to specify a file!")
- }
- filePath := os.Args[1]
- file, err := os.Open(filePath)
- if err != nil {
- log.Fatalf("Failed to open %s!\n", filePath)
- }
- machines := readInput(file)
- fmt.Println("Part1:", solve(machines, 100, 0, 0))
- fmt.Println("Part2:", solve(machines, -1, 10000000000000, 100000000000))
- }
|