Submission #6615268


Source Code Expand

;; -*- coding: utf-8 -*-
(eval-when (:compile-toplevel :load-toplevel :execute)
  (defparameter OPT
    #+swank '(optimize (speed 3) (safety 2))
    #-swank '(optimize (speed 3) (safety 0) (debug 0)))
  #+swank (ql:quickload '(:cl-debug-print :fiveam))
  #-swank (set-dispatch-macro-character #\# #\> (lambda (s c p) (declare (ignore c p)) (read s nil (values) t))))
#+swank (cl-syntax:use-syntax cl-debug-print:debug-print-syntax)
#-swank (disable-debugger) ; for CS Academy

;; BEGIN_INSERTED_CONTENTS
;;;
;;; Memoization macro
;;;

;; TODO: detailed documentation

;; Usage example:
;; (with-cache (:hash-table :test #'equal :key #'cons)
;;   (defun ...))
;; (with-cache (:array (10 10 * 10) :initial-element -1 :element-type 'fixnum)
;;   (defun foo (a b c d) ...)) ; => C is ignored.
;; (with-cache (:array (10 10) :initial-element -1 :element-type 'fixnum :debug t)
;;   (defun foo (x y) ...)) ; executes with trace of foo

;; FIXME: *RECURSION-DEPTH* should be included within the macro.
(declaim (type (integer 0 #.most-positive-fixnum) *recursion-depth*))
(defparameter *recursion-depth* 0)

(defmacro with-cache (cache-attribs def-form)
  (let* ((cache-attribs (if (atom cache-attribs) (list cache-attribs) cache-attribs))
         (cache-type (first cache-attribs))
         (dimensions-with-* (when (eql cache-type :array) (second cache-attribs)))
         (dimensions (remove '* dimensions-with-*))
         (rank (length dimensions))
         (rest-attribs (ecase cache-type
                         (:hash-table (cdr cache-attribs))
                         (:array (cddr cache-attribs))))
         (key (prog1 (getf rest-attribs :key) (remf rest-attribs :key)))
         (debug (prog1 (getf rest-attribs :debug) (remf rest-attribs :debug)))
         (cache-form (case cache-type
                       (:hash-table `(make-hash-table ,@rest-attribs))
                       (:array `(make-array (list ,@dimensions) ,@rest-attribs))))
         (initial-element (when (eql cache-type :array)
                            (assert (member :initial-element rest-attribs))
                            (getf rest-attribs :initial-element))))
    (let ((cache (gensym))
          (value (gensym))
	  (present-p (gensym))
          (name-alias (gensym))
	  (args-lst (gensym))
          (indices (loop repeat rank collect (gensym))))
      (labels ((debug (name args obj)
                 (let ((value (gensym)))
                   (if debug
                       `(progn
                          (format t "~&~A~A: (~A ~{~A~^ ~}) =>"
                                  (make-string *recursion-depth*
                                               :element-type 'base-char
                                               :initial-element #\ )
                                  *recursion-depth*
                                  ',name
                                  (list ,@args))
                          (let ((,value (let ((*recursion-depth* (1+ *recursion-depth*)))
                                          ,obj)))
                            (format t "~&~A~A: (~A ~{~A~^ ~}) => ~A"
                                    (make-string *recursion-depth*
                                                 :element-type 'base-char
                                                 :initial-element #\ )
                                    *recursion-depth*
                                    ',name
                                    (list ,@args)
                                    ,value)
                            ,value))
                       obj)))
               (make-cache-check-form (cache-type name args)
                 (debug name
                        args
                        (case cache-type
                          (:hash-table
                           `(let ((,args-lst (funcall ,(or key #'list) ,@args)))
                              (multiple-value-bind (,value ,present-p)
                                  (gethash ,args-lst ,cache)
                                (if ,present-p
                                    ,value
                                    (setf (gethash ,args-lst ,cache)
                                          (,name-alias ,@args))))))
                          (:array
                           (let ((memoized-args (loop for dimension in dimensions-with-*
                                                      for arg in args
                                                      unless (eql dimension '*)
                                                      collect arg)))
                             (if key
                                 `(multiple-value-bind ,indices
                                      (funcall ,key ,@memoized-args)
                                    (let ((,value (aref ,cache ,@indices)))
                                      (if (eql ,initial-element ,value)
                                          (setf (aref ,cache ,@indices)
                                                (,name-alias ,@args))
                                          ,value)))
                                 `(let ((,value (aref ,cache ,@memoized-args)))
                                    (if (eql ,initial-element ,value)
                                        (setf (aref ,cache ,@memoized-args)
                                              (,name-alias ,@args))
                                        ,value))))))))
               (make-reset-form (cache-type)
                 (case cache-type
                   (:hash-table `(setf ,cache (make-hash-table ,@rest-attribs)))
                   (:array `(prog1 nil
                              (fill (array-storage-vector ,cache) ,initial-element)))))
               (make-reset-name (name)
                 (intern (format nil "RESET-~A" (symbol-name name))))
               (extract-declarations (body)
                 (remove-if-not (lambda (form) (eql 'declare (car form))) body)))
        (ecase (car def-form)
          ((defun)
           (destructuring-bind (_ name args &body body) def-form
             (declare (ignore _))
             `(let ((,cache ,cache-form))
                (defun ,(make-reset-name name) () ,(make-reset-form cache-type))
                (defun ,name ,args
                  ,@(extract-declarations body)
                  (labels ((,name-alias ,args ,@body))
                    (declare (inline ,name-alias))
                    ,(make-cache-check-form cache-type name args))))))
          ((nlet #+sbcl sb-int:named-let)
           (destructuring-bind (_ name bindings &body body) def-form
             (declare (ignore _))
             `(let ((,cache ,cache-form))
                (,(car def-form) ,name ,bindings
                 ,@(extract-declarations body)
                 ,(let ((args (mapcar (lambda (x) (if (atom x) x (car x))) bindings)))
                    `(labels ((,name-alias ,args ,@body))
                       (declare (inline ,name-alias))
                       ,(make-cache-check-form cache-type name args)))))))
          ((labels flet)
           (destructuring-bind (_ definitions &body labels-body) def-form
             (declare (ignore _))
             (destructuring-bind (name args &body body) (car definitions)
               `(let ((,cache ,cache-form))
                  (,(car def-form)
                   ((,(make-reset-name name) () ,(make-reset-form cache-type))
                    (,name ,args
                           ,@(extract-declarations body)
                           (labels ((,name-alias ,args ,@body))
                             (declare (inline ,name-alias))
                             ,(make-cache-check-form cache-type name args)))
                    ,@(cdr definitions))
                   (declare (ignorable #',(make-reset-name name)))
                   ,@labels-body))))))))))

(defmacro dbg (&rest forms)
  #+swank
  (if (= (length forms) 1)
      `(format *error-output* "~A => ~A~%" ',(car forms) ,(car forms))
      `(format *error-output* "~A => ~A~%" ',forms `(,,@forms)))
  #-swank (declare (ignore forms)))

(defmacro define-int-types (&rest bits)
  `(progn
     ,@(mapcar (lambda (b) `(deftype ,(intern (format nil "UINT~A" b)) () '(unsigned-byte ,b))) bits)
     ,@(mapcar (lambda (b) `(deftype ,(intern (format nil "INT~A" b)) () '(signed-byte ,b))) bits)))
(define-int-types 2 4 7 8 15 16 31 32 62 63 64)

(declaim (inline println))
(defun println (obj &optional (stream *standard-output*))
  (let ((*read-default-float-format* 'double-float))
    (prog1 (princ obj stream) (terpri stream))))

(defconstant +mod+ 1000000007)

;; Body

(defun main ()
  (declare #.OPT)
  (let* ((n (read))
         (m (read))
         (probs (make-array (list m n) :element-type 'double-float :initial-element 0d0))
         (costs (make-array m :element-type 'double-float))
         (universe (- (expt 2 n) 1)))
    (declare ((integer 0 10) n m))
    (dotimes (i m)
      (let ((c (read))
            (cost (read)))
        (setf (aref costs i) (float cost 1d0))
        (dotimes (j c)
          (let ((idol (- (read) 1))
                (p (read)))
            (setf (aref probs i idol) (/ p 100d0))))))
    (with-cache (:array ((expt 2 n)) :element-type 'double-float :initial-element -1d0)
      (labels ((recur (s)
                 (if (= s universe)
                     0d0
                     (let ((res most-positive-double-float))
                       (declare (double-float res))
                       (dotimes (lot m)
                         (let ((denom (- 1d0 (loop for i below n
                                                   when (logbitp i s)
                                                   sum (aref probs lot i)
                                                   of-type double-float))))
                           (declare (double-float denom))
                           (unless (< denom 1d-8)
                             (setf res
                                   (min res
                                        (/ (+ (aref costs lot)
                                              (loop for i below n
                                                    unless (logbitp i s)
                                                    sum (* (recur (dpb 1 (byte 1 i) s))
                                                           (aref probs lot i))
                                                    of-type double-float))
                                           denom))))))
                       res))))
        (println (recur 0))))))

#-swank (main)

Submission Info

Submission Time
Task C - ソーシャルゲーム
User sansaqua
Language Common Lisp (SBCL 1.1.14)
Score 100
Code Size 10749 Byte
Status AC
Exec Time 254 ms
Memory 29028 KiB

Judge Result

Set Name A B C D E all
Score / Max Score 10 / 10 10 / 10 10 / 10 20 / 20 20 / 20 30 / 30
Status
AC × 4
AC × 11
AC × 20
AC × 39
AC × 12
AC × 59
Set Name Test Cases
A test_01_ABCDEF.txt, test_02_ABCDEF.txt, test_03_ABCDEF.txt, test_12_ABCDEF.txt
B test_01_ABCDEF.txt, test_02_ABCDEF.txt, test_03_ABCDEF.txt, test_04_BCDF.txt, test_05_BCDF.txt, test_06_BCDF.txt, test_07_BCDF.txt, test_08_BCDF.txt, test_09_BCDF.txt, test_11_BCDF.txt, test_12_ABCDEF.txt
C test_01_ABCDEF.txt, test_02_ABCDEF.txt, test_03_ABCDEF.txt, test_04_BCDF.txt, test_05_BCDF.txt, test_06_BCDF.txt, test_07_BCDF.txt, test_08_BCDF.txt, test_09_BCDF.txt, test_10_CF.txt, test_11_BCDF.txt, test_12_ABCDEF.txt, test_13_CF.txt, test_14_CDF.txt, test_15_CDF.txt, test_16_CF.txt, test_17_CF.txt, test_18_CF.txt, test_25_CDF.txt, test_26_CDF.txt
D test_01_ABCDEF.txt, test_02_ABCDEF.txt, test_03_ABCDEF.txt, test_04_BCDF.txt, test_05_BCDF.txt, test_06_BCDF.txt, test_07_BCDF.txt, test_08_BCDF.txt, test_09_BCDF.txt, test_11_BCDF.txt, test_12_ABCDEF.txt, test_14_CDF.txt, test_15_CDF.txt, test_19_DEF.txt, test_20_DF.txt, test_21_DF.txt, test_22_DF.txt, test_23_DEF.txt, test_24_DF.txt, test_25_CDF.txt, test_26_CDF.txt, test_27_DEF.txt, test_28_DF.txt, test_29_DF.txt, test_30_DF.txt, test_31_DEF.txt, test_32_DF.txt, test_33_DF.txt, test_34_DF.txt, test_35_DEF.txt, test_36_DF.txt, test_37_DF.txt, test_38_DF.txt, test_39_DEF.txt, test_40_DF.txt, test_41_DF.txt, test_42_DF.txt, test_44_DF.txt, test_52_DF.txt
E test_01_ABCDEF.txt, test_02_ABCDEF.txt, test_03_ABCDEF.txt, test_12_ABCDEF.txt, test_19_DEF.txt, test_23_DEF.txt, test_27_DEF.txt, test_31_DEF.txt, test_35_DEF.txt, test_39_DEF.txt, test_49_EF.txt, test_51_EF.txt
all 00_sample_01_F.txt, 00_sample_02_F.txt, 00_sample_03_F.txt, 00_sample_04_F.txt, 00_sample_05_F.txt, test_01_ABCDEF.txt, test_02_ABCDEF.txt, test_03_ABCDEF.txt, test_04_BCDF.txt, test_05_BCDF.txt, test_06_BCDF.txt, test_07_BCDF.txt, test_08_BCDF.txt, test_09_BCDF.txt, test_10_CF.txt, test_11_BCDF.txt, test_12_ABCDEF.txt, test_13_CF.txt, test_14_CDF.txt, test_15_CDF.txt, test_16_CF.txt, test_17_CF.txt, test_18_CF.txt, test_19_DEF.txt, test_20_DF.txt, test_21_DF.txt, test_22_DF.txt, test_23_DEF.txt, test_24_DF.txt, test_25_CDF.txt, test_26_CDF.txt, test_27_DEF.txt, test_28_DF.txt, test_29_DF.txt, test_30_DF.txt, test_31_DEF.txt, test_32_DF.txt, test_33_DF.txt, test_34_DF.txt, test_35_DEF.txt, test_36_DF.txt, test_37_DF.txt, test_38_DF.txt, test_39_DEF.txt, test_40_DF.txt, test_41_DF.txt, test_42_DF.txt, test_43_F.txt, test_44_DF.txt, test_45_F.txt, test_46_F.txt, test_47_F.txt, test_48_F.txt, test_49_EF.txt, test_50_F.txt, test_51_EF.txt, test_52_DF.txt, test_53_F.txt, test_54_F.txt
Case Name Status Exec Time Memory
00_sample_01_F.txt AC 254 ms 29028 KiB
00_sample_02_F.txt AC 73 ms 16872 KiB
00_sample_03_F.txt AC 71 ms 16864 KiB
00_sample_04_F.txt AC 73 ms 16864 KiB
00_sample_05_F.txt AC 73 ms 16868 KiB
test_01_ABCDEF.txt AC 71 ms 16868 KiB
test_02_ABCDEF.txt AC 71 ms 16864 KiB
test_03_ABCDEF.txt AC 71 ms 16868 KiB
test_04_BCDF.txt AC 72 ms 16868 KiB
test_05_BCDF.txt AC 71 ms 16868 KiB
test_06_BCDF.txt AC 71 ms 16868 KiB
test_07_BCDF.txt AC 72 ms 16868 KiB
test_08_BCDF.txt AC 72 ms 16868 KiB
test_09_BCDF.txt AC 71 ms 16872 KiB
test_10_CF.txt AC 71 ms 16868 KiB
test_11_BCDF.txt AC 72 ms 16868 KiB
test_12_ABCDEF.txt AC 72 ms 16868 KiB
test_13_CF.txt AC 71 ms 16868 KiB
test_14_CDF.txt AC 71 ms 16868 KiB
test_15_CDF.txt AC 71 ms 16872 KiB
test_16_CF.txt AC 71 ms 16872 KiB
test_17_CF.txt AC 72 ms 16868 KiB
test_18_CF.txt AC 72 ms 16868 KiB
test_19_DEF.txt AC 72 ms 16868 KiB
test_20_DF.txt AC 72 ms 16872 KiB
test_21_DF.txt AC 72 ms 16868 KiB
test_22_DF.txt AC 71 ms 16864 KiB
test_23_DEF.txt AC 71 ms 16864 KiB
test_24_DF.txt AC 71 ms 16864 KiB
test_25_CDF.txt AC 72 ms 16872 KiB
test_26_CDF.txt AC 73 ms 16872 KiB
test_27_DEF.txt AC 72 ms 16868 KiB
test_28_DF.txt AC 74 ms 16872 KiB
test_29_DF.txt AC 72 ms 16868 KiB
test_30_DF.txt AC 73 ms 16868 KiB
test_31_DEF.txt AC 72 ms 16868 KiB
test_32_DF.txt AC 75 ms 16868 KiB
test_33_DF.txt AC 72 ms 16872 KiB
test_34_DF.txt AC 72 ms 16868 KiB
test_35_DEF.txt AC 72 ms 16868 KiB
test_36_DF.txt AC 72 ms 16868 KiB
test_37_DF.txt AC 72 ms 16872 KiB
test_38_DF.txt AC 71 ms 16868 KiB
test_39_DEF.txt AC 72 ms 16864 KiB
test_40_DF.txt AC 72 ms 16864 KiB
test_41_DF.txt AC 71 ms 16864 KiB
test_42_DF.txt AC 72 ms 16864 KiB
test_43_F.txt AC 72 ms 16868 KiB
test_44_DF.txt AC 71 ms 16868 KiB
test_45_F.txt AC 72 ms 16868 KiB
test_46_F.txt AC 71 ms 16864 KiB
test_47_F.txt AC 72 ms 16864 KiB
test_48_F.txt AC 72 ms 16868 KiB
test_49_EF.txt AC 72 ms 16868 KiB
test_50_F.txt AC 73 ms 16868 KiB
test_51_EF.txt AC 72 ms 16872 KiB
test_52_DF.txt AC 71 ms 16868 KiB
test_53_F.txt AC 71 ms 16868 KiB
test_54_F.txt AC 73 ms 16864 KiB