I am trying to learn CLOS and bumped into this surprise. I have those 3 action methods that seemingly incorrectly match on my arguments. When I run tryme function, why the action marked with the last argument being ':c' does not trigger the expected error? What am I doing wrong? Or is it an SBCL bug?
(eval-when (:compile-toplevel :load-toplevel :execute)
(ql:quickload '(alexandria serapeum defclass-std)))
;; (load "~/AAA//clos-turnstile.lisp")
(in-package "CL-USER")
(shadowing-import 'defclass-std::defclass/std)
;;; -----------------------------------------------------------------------------------------
(defclass/std current-state () ())
(defclass/std locked (current-state) ())
(defclass/std unlocked (current-state) ())
(defclass/std input () ())
(defclass/std icoin (input) ())
(defclass/std ipush (input) ())
(defclass/std turnstile ()
((state :std (make-instance 'locked))))
(defgeneric action (turnstile state i msg)
(:documentation "action for our FSM"))
(defmethod action ((turnstile turnstile) (any-state T) (any-input T) (msg T))
(error "0 unmatched action for ~S ~S ~a" any-state any-input msg))
(defmethod action ((turnstile turnstile) (locked current-state) (i icoin) (msg T))
(warn "1 lock coin unlock ~S" msg))
(defmethod action ((turnstile turnstile) (unlocked current-state) (i ipush) (msg T))
(warn "2 unlock push lock ~s" msg))
(defparameter *turnstile* (make-instance 'turnstile))
(defun show-me ()
(format t "=== ~S~%" *turnstile*))
(defun tryme ()
(show-me)
(action *turnstile* (make-instance 'locked) (make-instance 'icoin) :a)
(show-me)
(action *turnstile* (make-instance 'unlocked) (make-instance 'ipush) :b)
(show-me)
;; why this does not give the error? I have swapped the argument types.
(action *turnstile* (make-instance 'unlocked) (make-instance 'icoin) :c)
(show-me)
(action *turnstile* (make-instance 'locked) (make-instance 'icoin) :d)
(show-me)
(action *turnstile* (make-instance 'unlocked) (make-instance 'ipush) :e)
(show-me)
*turnstile*)
Suspicion
I suspect I should have:
(locked locked)
instead of (locked current-state)
in the method definition, but how do I avoid such mistakes?
With the modified arguments of the method definitions, I get the expected error.