2.rkt 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687
  1. #lang racket
  2. (require "../lib/utils.rkt")
  3. (define lines
  4. (call-with-input-file "input"
  5. (λ (fp) (get-lines fp))))
  6. (set! lines (list->vector lines))
  7. (define (transpose linesvec)
  8. (define lines (vector->list linesvec))
  9. (define width (string-length (vector-ref linesvec 0)))
  10. (define new-lines (make-vector width))
  11. (let loop ((i 0))
  12. (if (>= i width)
  13. new-lines
  14. (let ()
  15. (define char-list (map (λ (l) (string-ref l i)) lines))
  16. (vector-set! new-lines i (list->string char-list))
  17. (loop (+ i 1))))))
  18. (define (get-rows-to-expand lines)
  19. (let loop ((i 0) (ret '()))
  20. (if (>= i (vector-length lines))
  21. (reverse (cons 9999999 ret))
  22. (let ()
  23. (define str (vector-ref lines i))
  24. (if (andmap (λ (c) (char=? #\. c))
  25. (string->list str))
  26. (loop (+ 1 i) (cons i ret))
  27. (loop (+ 1 i) ret))))))
  28. (define rows-to-expand (get-rows-to-expand lines))
  29. (define columns-to-expand (get-rows-to-expand (transpose lines)))
  30. (define height (vector-length lines))
  31. (define width (string-length (vector-ref lines 0)))
  32. (define expand-param 1000000)
  33. (define (build-expand-map num rows-to-expand)
  34. (define map (make-vector num))
  35. (let loop ((i 0) (cur 0) (rows rows-to-expand))
  36. (if (>= i num)
  37. map
  38. (begin
  39. (vector-set! map i cur)
  40. (if (= i (car rows))
  41. (loop (+ 1 i) (+ expand-param cur) (cdr rows))
  42. (loop (+ 1 i) (+ 1 cur) rows))))))
  43. (define row-expand-map (build-expand-map height rows-to-expand))
  44. (define column-expand-map (build-expand-map width columns-to-expand))
  45. (define (get-galaxies universe)
  46. (define (char-at x y) (string-ref (vector-ref universe y) x))
  47. (let loop ((x 0) (y 0) (ret '()))
  48. (if (>= y height)
  49. (list->vector (reverse ret))
  50. (if (>= x width)
  51. (loop 0 (+ y 1) ret)
  52. (if (char=? #\# (char-at x y))
  53. (loop (+ x 1) y (cons (cons x y) ret))
  54. (loop (+ x 1) y ret))))))
  55. (define galaxies (get-galaxies lines))
  56. (define (gen-pairs n)
  57. (let loop ((i 0) (j 1) (ret '()))
  58. (if (>= i n)
  59. (reverse ret)
  60. (if (>= j n)
  61. (loop (+ 1 i) (+ 2 i) ret)
  62. (loop i (+ 1 j) (cons (cons i j) ret))))))
  63. (define (distance n1 n2)
  64. (define x1 (vector-ref column-expand-map (car (vector-ref galaxies n1))))
  65. (define x2 (vector-ref column-expand-map (car (vector-ref galaxies n2))))
  66. (define y1 (vector-ref row-expand-map (cdr (vector-ref galaxies n1))))
  67. (define y2 (vector-ref row-expand-map (cdr (vector-ref galaxies n2))))
  68. (+ (abs (- x1 x2)) (abs (- y1 y2))))
  69. (apply
  70. +
  71. (map (λ (x) (distance (car x) (cdr x)))
  72. (gen-pairs (vector-length galaxies))))