123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143 |
- package main
- import (
- "bufio"
- "fmt"
- "log"
- "os"
- "strconv"
- )
- func readInput(file *os.File) []int {
- scanner := bufio.NewScanner(file)
- var numbers []int
- for scanner.Scan() {
- line := scanner.Text()
- if line == "" {
- break
- }
- number, err := strconv.Atoi(line)
- if err != nil {
- log.Fatalf("Can't parse a number %s: %s", line, err)
- }
- numbers = append(numbers, number)
- }
- return numbers
- }
- func calculate(number int) int {
- a := number * 64
- number = (number ^ a) % 16777216
- b := number / 32
- number = (number ^ b) % 16777216
- c := number * 2048
- number = (number ^ c) % 16777216
- return number
- }
- func generateNumber(number int, iterations int) (int, []int) {
- lastDigits := make([]int, iterations+1)
- lastDigits[0] = number % 10
- for i := 0; i < iterations; i++ {
- number = calculate(number)
- lastDigits[i+1] = number % 10
- }
- return number, lastDigits
- }
- func part1(numbers []int, iterations int) (int, [][]int) {
- var result int
- var allLastDigits [][]int
- for _, number := range numbers {
- newNumber, lastDigits := generateNumber(number, iterations)
- result += newNumber
- allLastDigits = append(allLastDigits, lastDigits)
- }
- return result, allLastDigits
- }
- func isDesiredSequence(sequence, desiredSequence []int) bool {
- if len(sequence) != len(desiredSequence) {
- return false
- }
- for i := range sequence {
- if sequence[i] != desiredSequence[i] {
- return false
- }
- }
- return true
- }
- func sequenceKey(sequence []int) string {
- if len(sequence) != 4 {
- return ""
- }
- return fmt.Sprintf("%d_%d_%d_%d", sequence[0], sequence[1], sequence[2], sequence[3])
- }
- func highestSum(allLastDigits [][]int, iterations int) int {
- var allSequences [][]int
- sums := make(map[string]int)
- for _, lastDigits := range allLastDigits {
- var sequence []int
- checked := make(map[string]bool)
- for i := 1; i < iterations; i++ {
- sequence = append(sequence, lastDigits[i]-lastDigits[i-1])
- if len(sequence) > 3 {
- lastFour := sequence[len(sequence)-4:]
- if lastFour[3] <= 0 {
- continue
- }
- key := sequenceKey(lastFour)
- if !checked[key] {
- sums[key] += lastDigits[i]
- checked[key] = true
- }
- }
- }
- allSequences = append(allSequences, sequence)
- }
- var highest int
- for _, value := range sums {
- if value > highest {
- highest = value
- }
- }
- return highest
- }
- 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)
- }
- numbers := readInput(file)
- part1Result, allLastDigits := part1(numbers, 2000)
- fmt.Println("Part1:", part1Result)
- fmt.Println("Part2:", highestSum(allLastDigits, 2001))
- }
|