code.go 2.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  1. package main
  2. import (
  3. "bufio"
  4. "fmt"
  5. "log"
  6. "os"
  7. )
  8. func readInput(file *os.File) []int {
  9. scanner := bufio.NewScanner(file)
  10. var numbers []int
  11. for scanner.Scan() {
  12. line := scanner.Text()
  13. if line == "" {
  14. continue
  15. }
  16. var current int
  17. n, err := fmt.Sscanf(line, "%d", &current)
  18. if n != 1 || err != nil {
  19. log.Fatal("Can't parse:", line, err)
  20. }
  21. numbers = append(numbers, current)
  22. }
  23. return numbers
  24. }
  25. func indexOf(numbers []int, number int) int {
  26. for i := range numbers {
  27. if numbers[i] == number {
  28. return i
  29. }
  30. }
  31. return -1
  32. }
  33. func establishNewIndex(size int, current int, value int) int {
  34. delta := current + value
  35. if delta <= 0 {
  36. delta = 0 - delta
  37. rest := delta % size
  38. return size - rest
  39. }
  40. return delta % size
  41. }
  42. func removeAt(numbers []int, index int) []int {
  43. return append(numbers[:index], numbers[index+1:]...)
  44. }
  45. func addAt(numbers []int, value int, index int) []int {
  46. if index >= len(numbers) {
  47. return append(numbers, value)
  48. }
  49. var temp []int
  50. temp = append(temp, numbers[:index]...)
  51. temp = append(temp, value)
  52. return append(temp, numbers[index:]...)
  53. }
  54. func mix(numbers []int) []int {
  55. size := len(numbers)
  56. edge := size - 1
  57. mixed := make([]int, size)
  58. copy(mixed, numbers)
  59. for i := range numbers {
  60. if numbers[i] == 0 {
  61. continue
  62. }
  63. currentIndex := indexOf(mixed, numbers[i])
  64. newIndex := establishNewIndex(edge, currentIndex, numbers[i])
  65. mixed = removeAt(mixed, currentIndex)
  66. mixed = addAt(mixed, numbers[i], newIndex)
  67. }
  68. return mixed
  69. }
  70. func part1(mixed []int) int {
  71. zeroIndex := indexOf(mixed, 0)
  72. result := 0
  73. size := len(mixed)
  74. for i := 1; i < 4; i++ {
  75. index := (zeroIndex + i*1000) % size
  76. result += mixed[index]
  77. }
  78. return result
  79. }
  80. func main() {
  81. if len(os.Args) < 2 {
  82. log.Fatal("You need to specify a file!")
  83. }
  84. filePath := os.Args[1]
  85. file, err := os.Open(filePath)
  86. if err != nil {
  87. log.Fatalf("Failed to open %s!\n", filePath)
  88. }
  89. numbers := readInput(file)
  90. mixed := mix(numbers)
  91. fmt.Println("Part1:", part1(mixed))
  92. }