Finish the stupid goddamn pots
author |
Steve Losh <steve@stevelosh.com> |
date |
Sun, 16 Dec 2018 19:28:52 -0500 (2018-12-17) |
parents |
a19c9e1fd077
|
children |
a2fa45383a67
|
branches/tags |
(none) |
files |
package.lisp src/2018/day-12.lisp src/utils.lisp |
Changes
--- a/package.lisp Sun Dec 16 18:30:05 2018 -0500
+++ b/package.lisp Sun Dec 16 19:28:52 2018 -0500
@@ -21,6 +21,7 @@
:manhattan-neighbors
:nth-digit
:unique
+ :positions-if
:ring
:ring-prev
--- a/src/2018/day-12.lisp Sun Dec 16 18:30:05 2018 -0500
+++ b/src/2018/day-12.lisp Sun Dec 16 19:28:52 2018 -0500
@@ -3,64 +3,95 @@
(named-readtables:in-readtable :interpol-syntax)
+;;;; Pots ---------------------------------------------------------------------
(defstruct pots data min max)
(defmethod print-object ((o pots) s)
(print-unreadable-object (o s :type t)
(format s "~D to ~D: ~A" (pots-min o) (pots-max o)
(iterate (for i :from (pots-min o) :to (pots-max o))
- (collect (if (zerop (gethash i (pots-data o) 0)) #\. #\#)
+ (collect (if (plusp (pot o i)) #\# #\.)
:result-type 'string)))))
+
+
+(defun-inline pot (pots i)
+ (if (hset-contains-p (pots-data pots) i)
+ 1
+ 0))
+
+(defun add-pot (pots i)
+ (hset-insert! (pots-data pots) i))
+
+(defun rem-pot (pots i)
+ (hset-remove! (pots-data pots) i))
+
+(defun surroundings (pots i)
+ (make-array 5
+ :element-type 'bit
+ :initial-contents (list (pot pots (- i 2))
+ (pot pots (- i 1))
+ (pot pots i)
+ (pot pots (+ i 1))
+ (pot pots (+ i 2)))))
+
+(defun score (pots)
+ (summation (hset-elements (pots-data pots))))
+
+
+;;;; Input Parsing ------------------------------------------------------------
(defun rune-bit (rune)
(ecase rune
(#\# 1)
(#\. 0)))
(defun runes-to-bits (runes)
- (map 'list #'rune-bit runes))
-
-(defun list-to-hash-table (list)
- (iterate (for value :in list)
- (for i :from 0)
- (when (plusp value)
- (collect-hash (i value) :test 'eq))))
+ (map 'bit-vector #'rune-bit runes))
-(defun surroundings-key (ll l x r rr)
- (declare (type bit ll l x r rr))
- (+ (* (expt 2 0) ll)
- (* (expt 2 1) l)
- (* (expt 2 2) x)
- (* (expt 2 3) r)
- (* (expt 2 4) rr)))
+(defun vector-to-hash-set (vector &key (test #'eql))
+ (iterate
+ (with result = (make-hash-set :test test))
+ (for value :in-vector vector :with-index i)
+ (when (plusp value)
+ (hset-insert! result i))))
(defun parse-initial-line (line)
(ppcre:register-groups-bind
(state)
(#?r"initial state: (\S+)" line)
- (list-to-hash-table (runes-to-bits state))))
+ (-<> state
+ runes-to-bits
+ (positions-if #'plusp <>)
+ (make-hash-set :initial-contents <>))))
(defun parse-rule (line)
(ppcre:register-groups-bind
(surroundings result)
(#?r"(\S+) => (\S)" line)
- (values (apply #'surroundings-key (runes-to-bits surroundings))
+ (values (runes-to-bits surroundings)
(rune-bit (aref result 0)))))
-(defun surroundings (state i)
- (let ((data (pots-data state)))
- (surroundings-key (gethash (- i 2) data 0)
- (gethash (- i 1) data 0)
- (gethash i data 0)
- (gethash (+ i 1) data 0)
- (gethash (+ i 2) data 0))))
+(defun read-problem (stream)
+ (let* ((initial (parse-initial-line (read-line stream)))
+ (state (prog1 (make-pots :data initial
+ :min (extremum (hset-elements initial) '<)
+ :max (extremum (hset-elements initial) '>))
+ (read-line stream)))
+ (rules (iterate
+ (for line :in-stream stream :using #'read-line)
+ (unless (string= "" line)
+ (for (values key result) = (parse-rule line))
+ (collect-hash (key result) :test #'equal)))))
+ (values state rules)))
-(defun tick (state rules)
- (with-slots (data min max) state
+
+;;;; Solve --------------------------------------------------------------------
+(defun tick (pots rules)
+ (with-slots (min max) pots
(iterate
(for i :from (- min 2) :to (+ max 2))
- (for current = (gethash i data 0))
- (for surroundings = (surroundings state i))
- (for next = (aref rules surroundings))
+ (for current = (pot pots i))
+ (for surroundings = (surroundings pots i))
+ (for next = (gethash surroundings rules))
(when (plusp next)
(minimizing i :into next-min)
(maximizing i :into next-max))
@@ -69,37 +100,38 @@
(collect i :into add)
(collect i :into rem)))
(finally
- (dolist (i add) (setf (gethash i data) 1))
- (dolist (i rem) (remhash i data))
+ (map nil (curry #'add-pot pots) add)
+ (map nil (curry #'rem-pot pots) rem)
(setf min next-min
max next-max)
- state))))
+ pots))))
(define-problem (2018 12) (data)
- (let* ((initial (parse-initial-line (read-line data)))
- (state (prog1 (make-pots :data initial
- :min (extremum (hash-table-keys initial) '<)
- :max (extremum (hash-table-keys initial) '>))
- (read-line data)))
- (rules (iterate
- (with rules = (make-array (expt 2 5) :initial-element 1))
- (for line :in-stream data :using #'read-line)
- (until (string= "" line))
- (for (values key result) = (parse-rule line))
- (setf (aref rules key) result)
- (finally (return rules)))))
- ;; Part 1 only
- (do-repeat 20
- (tick state rules))
- (summation (hash-table-keys (pots-data state)))))
+ (multiple-value-bind (pots rules) (read-problem data)
+ (values
+ (progn
+ (do-repeat 20
+ (tick pots rules))
+ (score pots))
+ (iterate
+ (for tick :from 20)
+ (format t "~%After ~D tick~:P:~%~A~%score: ~D~%> " tick pots (score pots))
+ (force-output)
+ (for input = (read-line))
+ (until (string= "q" input))
+ (tick pots rules)))))
-;; (defun part-2 ()
-;; (progn (dotimes (i (- 500000 20))
-;; (when (dividesp i 1000)
-;; (pr i))
-;; (tick state rules))
-;; (summation (hash-table-keys (pots-data state)))))
+(defun part-2 ()
+ (let* ((score-per-tick 20)
+ (starting-tick 350)
+ (starting-value 7508)
+ (ticks (- 50000000000 starting-tick)))
+ (+ starting-value (* score-per-tick ticks))))
+
+;;;; Test ---------------------------------------------------------------------
(1am:test test-2018/12
- (multiple-value-bind (part1) (run)
- (1am:is (= 1733 part1))))
+ ;; can't really test noninteractively :(
+ ;; (multiple-value-bind (part1) (run)
+ ;; (1am:is (= 1733 part1)))
+ (values))
--- a/src/utils.lisp Sun Dec 16 18:30:05 2018 -0500
+++ b/src/utils.lisp Sun Dec 16 19:28:52 2018 -0500
@@ -337,3 +337,28 @@
(if (or (= x last-col) (= y last-row))
0
(- (aref image (1+ x) (1+ y))))))))))
+
+
+(defun positions-if (predicate sequence &key (start 0) end key)
+ "Return a fresh list of all positions in `sequence` that satisfy `predicate`.
+
+ Like `cl:position-if`, but returns a list of all the results.
+
+ Example:
+
+ (positions-if #'upper-case-p \"aBCdeF\")
+ ; =>
+ (1 2 5)
+
+ "
+ (let ((pos start))
+ (nreverse (reduce (lambda (result value)
+ (prog1 (if (funcall predicate value)
+ (cons pos result)
+ result)
+ (incf pos)))
+ sequence
+ :start start
+ :end end
+ :key key
+ :initial-value nil))))