aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--2022/puzzle-11.lisp116
1 files changed, 116 insertions, 0 deletions
diff --git a/2022/puzzle-11.lisp b/2022/puzzle-11.lisp
new file mode 100644
index 0000000..1a55a96
--- /dev/null
+++ b/2022/puzzle-11.lisp
@@ -0,0 +1,116 @@
+(load "lib")
+
+(defparameter input (uiop:read-file-string "11-input"))
+(defparameter lines (get-lines input))
+
+(defun lines->segments (lines)
+ (--> (split-by-completely lines "" #'string=)
+ (mapcar (lambda (x)
+ (--> (loop for i in x
+ collect
+ (string-trim '(#\Space) i))))
+ it)))
+
+(defun segment->monkey (segment)
+ (destructuring-bind (name starting op test clause-1 clause-2) segment
+ (declare (ignore name))
+ `(:items
+ ,(--> (subseq starting 16)
+ (string-to-clist it)
+ (split-by-completely it #\,)
+ (mapcar (lambda (lst)
+ (--> (clist-to-string lst)
+ (string-trim '(#\Space) it)
+ (parse-integer it)))
+ it))
+ :inspect
+ ,(--> (subseq op 21)
+ (format nil "(~a old)" it)
+ (with-input-from-string (s it)
+ (read s))
+ (eval `(lambda (old) ,it)))
+ :test
+ ,(let ((test (parse-integer (subseq test 19)))
+ (clause-1 (parse-integer (subseq clause-1 25)))
+ (clause-2 (parse-integer (subseq clause-2 26))))
+ (lambda (n)
+ (if (= (mod n test) 0)
+ clause-1
+ clause-2)))
+ :divisor ,(parse-integer (subseq test 19)))))
+
+(defmacro get-monkey (monkeys ind)
+ `(nth ,ind ,monkeys))
+
+(defmacro get-monkey-items (monkeys ind)
+ `(getf (get-monkey ,monkeys ,ind) :items))
+
+(defun monkey-increment-alist (monkey-alist monkey-ind)
+ (setf (cdr (assoc monkey-ind monkey-alist))
+ (+ 1 (cdr (assoc monkey-ind monkey-alist)))))
+
+(defun monkey-send-item (monkeys item recipient)
+ (setf (get-monkey-items monkeys recipient)
+ (append (get-monkey-items monkeys recipient)
+ (list item))))
+
+(defun monkey-process-item (inspect item)
+ (floor (/ (funcall inspect item) 3)))
+
+(defun monkey-do-turn (monkeys monkey-alist monkey-ind)
+ (let* ((monkey (nth monkey-ind monkeys))
+ (items (getf monkey :items))
+ (inspect (getf monkey :inspect))
+ (test (getf monkey :test)))
+ (loop for item in items
+ for proper-item = (monkey-process-item inspect item)
+ for recipient = (funcall test proper-item)
+ do (monkey-send-item monkeys proper-item recipient)
+ do (monkey-increment-alist monkey-alist monkey-ind))
+ (setf (get-monkey-items monkeys monkey-ind) nil)
+ monkeys))
+
+(defun monkeys-do-round (monkeys monkey-alist)
+ (loop for monkey in monkeys
+ for i from 0
+ do (monkey-do-turn monkeys monkey-alist i)))
+
+(defun monkey-business (monkey-alist)
+ (--> (lambda (x y) (> (cdr x) (cdr y)))
+ (sort monkey-alist it)
+ (subseq it 0 2)
+ (mapcar #'cdr it)
+ (reduce #'* it)))
+
+(let* ((monkeys (--> (lines->segments lines)
+ (mapcar #'segment->monkey it)))
+ (monkey-alist (loop for monkey in monkeys
+ for i from 0
+ collect `(,i . 0))))
+
+ (loop for i from 1 to 20
+ do (monkeys-do-round monkeys monkey-alist))
+
+ (format t "Round 1: ~a~%" (monkey-business monkey-alist)))
+
+(defvar *monkey-divisor* nil)
+
+(defun monkey-process-item (inspect item)
+ (mod (funcall inspect item) *monkey-divisor*))
+
+(let* ((monkeys (--> (lines->segments lines)
+ (mapcar #'segment->monkey it)))
+ (monkey-alist (loop for monkey in monkeys
+ for i from 0
+ collect `(,i . 0))))
+
+ ;; To ensure we don't go too hard.
+ (setq *monkey-divisor* (reduce #'*
+ (loop for monkey in monkeys
+ for divisor = (getf monkey :divisor)
+ collect divisor)))
+
+ (loop for i from 1 to 10000
+ do (monkeys-do-round monkeys monkey-alist))
+
+ (format t "Round 2: ~a~%" (monkey-business monkey-alist)))