main.go 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. package main
  2. import (
  3. "bufio"
  4. "errors"
  5. "fmt"
  6. "io"
  7. "log"
  8. "os"
  9. )
  10. const MAX_LOOP = 1_000_000
  11. func trim(bs []byte) []byte {
  12. last := len(bs) - 1
  13. if last >= 0 && bs[last] == '\n' {
  14. return bs[:last]
  15. }
  16. return bs
  17. }
  18. func main() {
  19. var dim int
  20. var rows [][]byte
  21. r := bufio.NewReader(os.Stdin)
  22. line, err := r.ReadBytes('\n')
  23. if err != nil {
  24. log.Fatalf("While reading input: %v", err)
  25. }
  26. line = trim(line)
  27. dim = len(line)
  28. rows = make([][]byte, dim)
  29. rows[0] = line
  30. for i := 1; i < dim; i++ {
  31. line, err = r.ReadBytes('\n')
  32. if errors.Is(err, io.EOF) {
  33. break
  34. }
  35. if err != nil {
  36. log.Fatalf("While reading input: %v", err)
  37. }
  38. rows[i] = trim(line)
  39. }
  40. diags := dim - 4 + dim - 4 + 1
  41. ctrs := 1 + dim + diags + diags
  42. bufs := make([]byte, ctrs*4)
  43. offset := 0
  44. horizontal := make([]*Counter, 1)
  45. horizontal[0] = new(Counter)
  46. horizontal[0].Init(bufs[offset : offset+4])
  47. offset += 4
  48. vertical := make([]*Counter, dim)
  49. for i := 0; i < dim; i++ {
  50. vertical[i] = new(Counter)
  51. vertical[i].Init(bufs[offset : offset+4])
  52. offset += 4
  53. }
  54. swDiag := make([]*Counter, diags)
  55. for i := 0; i < diags; i++ {
  56. swDiag[i] = new(Counter)
  57. swDiag[i].Init(bufs[offset : offset+4])
  58. offset += 4
  59. }
  60. seDiag := make([]*Counter, diags)
  61. for i := 0; i < diags; i++ {
  62. seDiag[i] = new(Counter)
  63. seDiag[i].Init(bufs[offset : offset+4])
  64. offset += 4
  65. }
  66. diagLag := 3
  67. for i := 0; i < dim; i++ {
  68. horizontal[0].Reset()
  69. for j := 0; j < dim; j++ {
  70. e := rows[i][j]
  71. horizontal[0].Event(e)
  72. vertical[j].Event(e)
  73. swIdx := i + j - diagLag
  74. if 0 <= swIdx && swIdx < diags {
  75. swDiag[swIdx].Event(e)
  76. }
  77. seIdx := j - i - (diagLag + 1 - dim)
  78. if 0 <= seIdx && seIdx < diags {
  79. seDiag[seIdx].Event(e)
  80. }
  81. }
  82. }
  83. sum := 0
  84. for _, c := range horizontal {
  85. sum += c.Count()
  86. }
  87. for _, c := range vertical {
  88. sum += c.Count()
  89. }
  90. for _, c := range swDiag {
  91. sum += c.Count()
  92. }
  93. for _, c := range seDiag {
  94. sum += c.Count()
  95. }
  96. fmt.Println(sum)
  97. }
  98. type Counter struct {
  99. cnt int
  100. buf []byte
  101. }
  102. func (c *Counter) Init(buf []byte) {
  103. c.buf = buf
  104. }
  105. func (c *Counter) Reset() {
  106. c.buf = c.buf[:0]
  107. }
  108. func (c *Counter) Event(e byte) {
  109. switch {
  110. case e == 'X' && string(c.buf) == "SAM" ||
  111. e == 'S' && string(c.buf) == "XMA":
  112. c.cnt++
  113. c.Reset()
  114. c.buf = append(c.buf, e)
  115. case e == 'M' && string(c.buf) == "SA" ||
  116. e == 'A' && string(c.buf) == "S" ||
  117. e == 'S' && string(c.buf) == "" ||
  118. e == 'A' && string(c.buf) == "XM" ||
  119. e == 'M' && string(c.buf) == "X" ||
  120. e == 'X' && string(c.buf) == "":
  121. c.buf = append(c.buf, e)
  122. default:
  123. c.Reset()
  124. c.buf = append(c.buf, e)
  125. }
  126. }
  127. func (c *Counter) Count() int {
  128. return c.cnt
  129. }