2.rkt 2.9 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495
  1. #lang racket
  2. (require "../lib/utils.rkt")
  3. (define lines
  4. (call-with-input-file "input"
  5. (λ (fp)
  6. (get-lines fp))))
  7. (define (unfold-pattern x)
  8. (append x (list #\?) x (list #\?) x (list #\?) x (list #\?) x))
  9. (define (unfold-num x)
  10. (append x x x x x))
  11. (define (parse-line line)
  12. (define line-splited (string-split line))
  13. (define arr-list (string->list (car line-splited)))
  14. (define conds (map string->number (string-split (cadr line-splited) ",")))
  15. (cons (reverse (cons #\. (reverse (unfold-pattern arr-list)))) (unfold-num conds)))
  16. (define (cont-lava scanned)
  17. (define nums (filter (λ (n) (not (= 0 n)))
  18. (map string-length (string-split scanned "."))))
  19. (if (null? nums)
  20. #f
  21. (car nums)))
  22. (define (can-match? scanned remain)
  23. (if (null? scanned)
  24. 'unknown
  25. (if (char=? #\# (car scanned))
  26. 'unknown
  27. (let ()
  28. (define num (cont-lava (list->string scanned)))
  29. (if (not num)
  30. 'unknown
  31. (if (null? remain)
  32. 'no
  33. (if (= num (car remain))
  34. 'yes
  35. 'no)))))))
  36. (define (check-terminal scanned remain)
  37. (define can-match (can-match? scanned remain))
  38. (if (and (eq? 'yes can-match)
  39. (null? (cdr remain)))
  40. 1
  41. (if (eq? 'no can-match)
  42. 0
  43. (if (null? remain)
  44. 1
  45. 0))))
  46. (define (possible-arr-num instance)
  47. (define conds (cdr instance))
  48. (define (impl scanned unscanned remain)
  49. (define can-match (can-match? scanned remain))
  50. (define (gen-next-scanned c lst)
  51. (if (and (null? lst) (char=? c #\.))
  52. '()
  53. (cons c lst)))
  54. (if (null? unscanned)
  55. (check-terminal scanned remain)
  56. (if (eq? can-match 'no)
  57. 0
  58. (let ()
  59. (define next-scanned
  60. (if (eq? 'yes can-match)
  61. '()
  62. scanned))
  63. (define next-remain
  64. (if (eq? 'yes can-match)
  65. (cdr remain)
  66. remain))
  67. (define head (car unscanned))
  68. (if (not (char=? head #\?))
  69. (impl (gen-next-scanned head next-scanned) (cdr unscanned) next-remain)
  70. (+ (impl (gen-next-scanned #\. next-scanned) (cdr unscanned) next-remain)
  71. (impl (gen-next-scanned #\# next-scanned) (cdr unscanned) next-remain)))))))
  72. (define memo (make-hash))
  73. (define original-impl impl)
  74. (define (memo-impl scanned unscanned remain)
  75. (if (null? scanned)
  76. (if (hash-has-key? memo (list unscanned remain))
  77. (hash-ref memo (list unscanned remain))
  78. (let ()
  79. (define ret (original-impl scanned unscanned remain))
  80. (hash-set! memo (list unscanned remain) ret)
  81. ret))
  82. (original-impl scanned unscanned remain)))
  83. (set! impl memo-impl)
  84. (impl '() (car instance) conds))
  85. (apply + (map possible-arr-num (map parse-line lines)))