Инициализация слотов на основе других ценностей слота в общих определениях класса объекта Lisp Objects

StackOverflow https://stackoverflow.com/questions/3620249

  •  26-09-2019
  •  | 
  •  

Вопрос

В определении моего класса я хочу инициализировать один слот на основе значения другого слота. Вот такая вещь, которую я хотел бы сделать:

(defclass my-class ()
  ((slot-1 :accessor my-class-slot-1 :initarg slot-1)
   (slot-2 :accessor my-class-slot-2 :initform (list slot-1))))

Однако это не скомпилируется:

1 compiler notes:

Unknown location:
  warning: 
    This variable is undefined:
      SLOT-1

  warning: 
    undefined variable: SLOT-1
    ==>
      (CONS UC-2::SLOT-1 NIL)


Compilation failed.

Есть ли способ сделать это?

Это было полезно?

Решение

Использовать initialize-instance :after документированный здесь

Другие советы

Вот ответ Дуга Курри расширился:

(defclass my-class ()
  ((slot-1 :accessor my-class-slot-1 :initarg :slot-1)
   (slot-2 :accessor my-class-slot-2)))

(defmethod initialize-instance :after 
           ((c my-class) &rest args)
  (setf (my-class-slot-2 c) 
        (list (my-class-slot-1 c))))

Вот звонок, показывающий, что он работает:

> (my-class-slot-2 (make-instance 'my-class :slot-1 "Bob"))
("Bob")

Видеть эта статья Больше подробностей.

(defparameter *self-ref* nil)


(defclass self-ref ()
  ()

  (:documentation "
Note that *SELF-REF* is not visible to code in :DEFAULT-INITARGS."))


(defmethod initialize-instance :around ((self-ref self-ref) &key)
  (let ((*self-ref* self-ref))
    (when (next-method-p)
      (call-next-method))))



(defclass my-class (self-ref)
  ((slot-1 :accessor slot-1-of :initarg :slot-1)
   (slot-2 :accessor slot-2-of
           :initform (slot-1-of *self-ref*))))




CL-USER> (let ((it (make-instance 'my-class :slot-1 42)))
           (values (slot-1-of it)
                   (slot-2-of it)))
42
42
CL-USER> 
Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top