mirror of
https://github.com/Threnklyn/advent-of-code-go.git
synced 2026-05-18 19:13:27 +02:00
137 lines
2.8 KiB
Go
137 lines
2.8 KiB
Go
package main
|
|
|
|
import (
|
|
"flag"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/alexchao26/advent-of-code-go/util"
|
|
)
|
|
|
|
func main() {
|
|
var part int
|
|
flag.IntVar(&part, "part", 1, "part 1 or 2")
|
|
flag.Parse()
|
|
fmt.Println("Running part", part)
|
|
|
|
if part == 1 {
|
|
ans := part1(util.ReadFile("./input.txt"))
|
|
fmt.Println("Output:", ans)
|
|
} else {
|
|
ans := part2(util.ReadFile("./input.txt"))
|
|
fmt.Println("Output:", ans)
|
|
}
|
|
}
|
|
|
|
func part1(input string) int {
|
|
comp := newComputerFromInput(input)
|
|
|
|
// keep track of all the indices of instructions that have run
|
|
// if it has already been run, break
|
|
ranInstructionsIndices := map[int]bool{}
|
|
for {
|
|
nextInst := comp.index
|
|
if ranInstructionsIndices[nextInst] {
|
|
break
|
|
}
|
|
ranInstructionsIndices[nextInst] = true
|
|
|
|
comp.step()
|
|
}
|
|
|
|
return comp.accumulator
|
|
}
|
|
|
|
func part2(input string) int {
|
|
comp := newComputerFromInput(input)
|
|
|
|
// iterate through instruction indices
|
|
for i := range comp.instructions {
|
|
// make new computer each time
|
|
newComputer := newComputerFromInput(input)
|
|
|
|
// flip this index's instruction if a jmp or nop
|
|
switch newComputer.instructions[i].instType {
|
|
case "jmp":
|
|
newComputer.instructions[i].instType = "nop"
|
|
case "nop":
|
|
newComputer.instructions[i].instType = "jmp"
|
|
case "acc":
|
|
continue
|
|
}
|
|
|
|
// run isInfiniteLoop check which returns final global value
|
|
if ans, isLoop := isInfiniteLoop(newComputer); !isLoop {
|
|
return ans
|
|
}
|
|
}
|
|
|
|
// this should never be hit
|
|
fmt.Println("ERROR: No terminating set of instructions found")
|
|
return -1
|
|
}
|
|
|
|
type instruction struct {
|
|
instType string
|
|
value int
|
|
}
|
|
|
|
func newComputerFromInput(input string) computer {
|
|
var instructions []instruction
|
|
|
|
lines := strings.Split(input, "\n")
|
|
for _, l := range lines {
|
|
inst := instruction{}
|
|
fmt.Sscanf(l, "%s %d", &inst.instType, &inst.value)
|
|
instructions = append(instructions, inst)
|
|
}
|
|
|
|
return computer{instructions: instructions}
|
|
}
|
|
|
|
type computer struct {
|
|
instructions []instruction
|
|
index int
|
|
accumulator int
|
|
}
|
|
|
|
func (c *computer) acc(val int) {
|
|
c.accumulator += val
|
|
c.index++
|
|
}
|
|
func (c *computer) jmp(val int) {
|
|
c.index += val
|
|
}
|
|
func (c *computer) nop(val int) {
|
|
c.index++
|
|
}
|
|
|
|
func (c *computer) step() {
|
|
switch inst := c.instructions[c.index]; inst.instType {
|
|
case "acc":
|
|
c.acc(inst.value)
|
|
case "jmp":
|
|
c.jmp(inst.value)
|
|
case "nop":
|
|
c.nop(inst.value)
|
|
}
|
|
}
|
|
|
|
func isInfiniteLoop(comp computer) (finalAccumulatorVal int, isLoop bool) {
|
|
ranInstructionsIndices := map[int]bool{}
|
|
for comp.index < len(comp.instructions) {
|
|
nextInst := comp.index
|
|
// is an infinite loop, return out
|
|
if ranInstructionsIndices[nextInst] {
|
|
return 0, true
|
|
}
|
|
ranInstructionsIndices[nextInst] = true
|
|
|
|
comp.step()
|
|
}
|
|
|
|
// instructions finished, return final accumulator & indicate it was not an
|
|
// infinite loop
|
|
return comp.accumulator, false
|
|
}
|