mirror of
https://github.com/Threnklyn/advent-of-code-go.git
synced 2026-05-18 19:13:27 +02:00
day 8 - lucky lucky kind inputs
This commit is contained in:
@@ -0,0 +1,145 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
_ "embed"
|
||||
"flag"
|
||||
"fmt"
|
||||
"strings"
|
||||
|
||||
"github.com/alexchao26/advent-of-code-go/mathy"
|
||||
"github.com/alexchao26/advent-of-code-go/util"
|
||||
)
|
||||
|
||||
//go:embed input.txt
|
||||
var input string
|
||||
|
||||
func init() {
|
||||
// do this in init (not main) so test file has same input
|
||||
input = strings.TrimRight(input, "\n")
|
||||
if len(input) == 0 {
|
||||
panic("empty input.txt file")
|
||||
}
|
||||
}
|
||||
|
||||
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(input)
|
||||
util.CopyToClipboard(fmt.Sprintf("%v", ans))
|
||||
fmt.Println("Output:", ans)
|
||||
} else {
|
||||
ans := part2(input)
|
||||
util.CopyToClipboard(fmt.Sprintf("%v", ans))
|
||||
fmt.Println("Output:", ans)
|
||||
}
|
||||
}
|
||||
|
||||
func part1(input string) int {
|
||||
steps, graph := parseInput(input)
|
||||
|
||||
location := "AAA"
|
||||
numOfSteps := 0
|
||||
for location != "ZZZ" {
|
||||
dir := steps[numOfSteps%len(steps)]
|
||||
|
||||
if dir == "L" {
|
||||
location = graph[location][0]
|
||||
} else {
|
||||
location = graph[location][1]
|
||||
}
|
||||
|
||||
numOfSteps++
|
||||
}
|
||||
|
||||
return numOfSteps
|
||||
}
|
||||
|
||||
func part2(input string) int {
|
||||
steps, graph := parseInput(input)
|
||||
|
||||
locations := []string{}
|
||||
for loc := range graph {
|
||||
if loc[2:3] == "A" {
|
||||
locations = append(locations, loc)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
brute force doesn't work... need to figure out cycle times of each starting location
|
||||
but they won't cycle just based on number of steps because of the weird L-R randomness
|
||||
|
||||
so we can only rely on the "full cycle" of all steps before it loops
|
||||
|
||||
- there are six starting locations
|
||||
|
||||
NOTE: BIG assumptions based on KIND inputs
|
||||
- assume that the Z-end locations will sync EXACTLY at the end of a cycle of steps
|
||||
- after further analyzing logs of the end of each cycle, the entry point VERY kindly deposits us
|
||||
at the very start of a cycle that will eventually end in a Z-end location
|
||||
AAA -> MLM -> ... -> XKZ -> MLM -> ... -> XKZ -> MLM -> ... -> XKZ -> MLM
|
||||
and this holds true for all six locations in my input
|
||||
Therefore the cycles are not offset by a particular number of steps at the start to get to the cycle
|
||||
such as START --> LOC1 --> LOC2 --> Start -> A
|
||||
^ |
|
||||
| v
|
||||
D <-- C
|
||||
this makes the maths fairly straight forward with just having to find the LCM (least common multiple)
|
||||
of all the cycle periods because that is when they will all sync up and land on a Z
|
||||
*/
|
||||
|
||||
numOfSteps := 0
|
||||
|
||||
locationCyclePeriods := []int{}
|
||||
for cycle := 0; len(locations) > 0; cycle++ {
|
||||
for _, dir := range steps {
|
||||
for i, loc := range locations {
|
||||
if dir == "L" {
|
||||
locations[i] = graph[loc][0]
|
||||
} else {
|
||||
locations[i] = graph[loc][1]
|
||||
}
|
||||
}
|
||||
numOfSteps++
|
||||
}
|
||||
|
||||
// if any location is at a z-end at the end of a cycle, record the cycle time
|
||||
// to do the final maths at the end
|
||||
newLocations := []string{}
|
||||
for _, loc := range locations {
|
||||
if loc[2:3] == "Z" {
|
||||
locationCyclePeriods = append(locationCyclePeriods, numOfSteps)
|
||||
} else {
|
||||
newLocations = append(newLocations, loc)
|
||||
}
|
||||
}
|
||||
locations = newLocations
|
||||
}
|
||||
|
||||
// combine all into an LCM (helper function added to mathy package)
|
||||
lcm := locationCyclePeriods[0]
|
||||
for i := 1; i < len(locationCyclePeriods); i++ {
|
||||
lcm = mathy.LeastCommonMultiple(lcm, locationCyclePeriods[i])
|
||||
}
|
||||
|
||||
return lcm
|
||||
}
|
||||
|
||||
func parseInput(input string) (steps []string, graph map[string][]string) {
|
||||
graph = map[string][]string{}
|
||||
|
||||
parts := strings.Split(input, "\n\n")
|
||||
steps = strings.Split(parts[0], "")
|
||||
|
||||
for _, line := range strings.Split(parts[1], "\n") {
|
||||
graph[line[0:3]] = []string{
|
||||
line[7:10],
|
||||
line[12:15],
|
||||
}
|
||||
}
|
||||
|
||||
return steps, graph
|
||||
}
|
||||
@@ -0,0 +1,74 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"testing"
|
||||
)
|
||||
|
||||
var example = `LLR
|
||||
|
||||
AAA = (BBB, BBB)
|
||||
BBB = (AAA, ZZZ)
|
||||
ZZZ = (ZZZ, ZZZ)`
|
||||
|
||||
func Test_part1(t *testing.T) {
|
||||
tests := []struct {
|
||||
name string
|
||||
input string
|
||||
want int
|
||||
}{
|
||||
{
|
||||
name: "example",
|
||||
input: example,
|
||||
want: 6,
|
||||
},
|
||||
{
|
||||
name: "actual",
|
||||
input: input,
|
||||
want: 18673,
|
||||
},
|
||||
}
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
if got := part1(tt.input); got != tt.want {
|
||||
t.Errorf("part1() = %v, want %v", got, tt.want)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
var examplePart2 = `LR
|
||||
|
||||
11A = (11B, XXX)
|
||||
11B = (XXX, 11Z)
|
||||
11Z = (11B, XXX)
|
||||
22A = (22B, XXX)
|
||||
22B = (22C, 22C)
|
||||
22C = (22Z, 22Z)
|
||||
22Z = (22B, 22B)
|
||||
XXX = (XXX, XXX)`
|
||||
|
||||
func Test_part2(t *testing.T) {
|
||||
tests := []struct {
|
||||
name string
|
||||
input string
|
||||
want int
|
||||
}{
|
||||
{
|
||||
name: "example",
|
||||
input: examplePart2,
|
||||
want: 6,
|
||||
},
|
||||
{
|
||||
name: "actual",
|
||||
input: input,
|
||||
want: 17972669116327,
|
||||
},
|
||||
}
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
if got := part2(tt.input); got != tt.want {
|
||||
t.Errorf("part2() = %v, want %v", got, tt.want)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,18 @@
|
||||
package mathy
|
||||
|
||||
func LeastCommonMultiple(i, j int) int {
|
||||
gcd := GreatestCommonMultiple(i, j)
|
||||
|
||||
return (i * j) / gcd
|
||||
}
|
||||
|
||||
func GreatestCommonMultiple(i, j int) int {
|
||||
if j > i {
|
||||
i, j = j, i
|
||||
}
|
||||
|
||||
if j == 0 {
|
||||
return i
|
||||
}
|
||||
return GreatestCommonMultiple(j, i%j)
|
||||
}
|
||||
Reference in New Issue
Block a user