2.rkt 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  1. #lang racket
  2. (require "../lib/utils.rkt")
  3. (define lines
  4. (call-with-input-file "input"
  5. (λ (fp) (get-lines fp))))
  6. (define (mat-copy mat)
  7. (list->vector (map string-copy (vector->list mat))))
  8. (define (flip-mat mat)
  9. (list->vector (reverse (map string-copy (vector->list mat)))))
  10. (define mat (list->vector lines))
  11. (define (char-at mat x y)
  12. (string-ref (vector-ref mat y) x))
  13. (define (set-mat! mat x y c)
  14. (string-set! (vector-ref mat y) x c))
  15. (define (move-stone mat x1 y1 x2 y2)
  16. (define t (char-at mat x1 y1))
  17. (set-mat! mat x1 y1 (char-at mat x2 y2))
  18. (set-mat! mat x2 y2 t))
  19. (define (find-new-pos mat x y)
  20. (let loop ((new-y y))
  21. (if (or (= new-y 0)
  22. (not (char=? #\. (char-at mat x (- new-y 1)))))
  23. new-y
  24. (loop (- new-y 1)))))
  25. (define height (vector-length mat))
  26. (define width (string-length (vector-ref mat 0)))
  27. (define (tilt old-mat)
  28. (define mat (mat-copy old-mat))
  29. (do ((y 0 (+ y 1)))
  30. ((>= y height) (void))
  31. (do ((x 0 (+ x 1)))
  32. ((>= x width) (void))
  33. (when (char=? #\O (char-at mat x y))
  34. (move-stone mat x y x (find-new-pos mat x y)))))
  35. mat)
  36. (define (count mat)
  37. (define sum 0)
  38. (do ((y 0 (+ y 1)))
  39. ((>= y height) (void))
  40. (do ((x 0 (+ x 1)))
  41. ((>= x width) (void))
  42. (when (char=? #\O (char-at mat x y))
  43. (set! sum (+ sum (- height y))))))
  44. sum)
  45. (define (tilt-north mat)
  46. (tilt mat))
  47. (define (tilt-south mat)
  48. (flip-mat (tilt (flip-mat mat))))
  49. (define (transpose linesvec)
  50. (define lines (vector->list linesvec))
  51. (define width (string-length (vector-ref linesvec 0)))
  52. (define new-lines (make-vector width))
  53. (let loop ((i 0))
  54. (if (>= i width)
  55. new-lines
  56. (let ()
  57. (define char-list (map (λ (l) (string-ref l i)) lines))
  58. (vector-set! new-lines i (list->string char-list))
  59. (loop (+ i 1))))))
  60. (define (tilt-west mat)
  61. (transpose (tilt (transpose mat))))
  62. (define (tilt-east mat)
  63. (transpose (flip-mat (tilt (flip-mat (transpose mat))))))
  64. (define mat-dict (make-hash))
  65. (define r-mat-dict (make-hash))
  66. (define (tilt-cycle mat)
  67. (tilt-east (tilt-south (tilt-west (tilt-north mat)))))
  68. ;; return (start . end)
  69. (define (find-loop mat)
  70. (let loop ((i 1) (cur-mat mat))
  71. (define m (tilt-cycle cur-mat))
  72. (if (hash-has-key? mat-dict (vector->list m))
  73. (cons (hash-ref mat-dict (vector->list m)) i)
  74. (let ()
  75. (hash-set! mat-dict (vector->list m) i)
  76. (hash-set! r-mat-dict i (vector->list m))
  77. (loop (+ 1 i) m)))))
  78. (define tmp (find-loop mat))
  79. (define loop-start (car tmp))
  80. (define loop-len (- (cdr tmp) (car tmp)))
  81. (define reduced-num
  82. (+ loop-start (modulo (- 1000000000 loop-start) loop-len)))
  83. (count (list->vector (hash-ref r-mat-dict reduced-num)))