Common Lisp 替代使用 类
Common Lisp alternative to using Classes
我想知道如何存储单个变量并在该变量上使用特定函数。我想知道是否有创建 class.
的替代方法
具体来说,我正在创建一个应用程序,我在其中存储一个时间值,表示从基准时间(例如,2000 年 1 月 1 日,00:00:00)开始经过的秒数。我想对该值执行操作,例如将其从秒数转换为特定时间或日期,或从日期转换为特定秒数。
我已经使用 class 完成了此操作,但它看起来很浪费。具体来说,每次我访问经过的秒数的存储值时,它看起来类似于 (time-time time)
,其中 time-time
是时间实例 time
.
的访问器
是否有更好的设计方法,也许没有 classes?
您可以像这样在词法作用域上尝试闭包:
(let ((time (get-universal-time)))
(defun set-time(tm)
(setf time tm))
(defun get-time()
time)
(defun get-time-in-some-other-format()
;; calculate return value based on 'time'
)
)
访问者名称
您可以在 CLOS 中以任何您喜欢的方式命名访问器。访问器函数可以称为 seconds
:
CL-USER 23 > (defclass mytime ()
((seconds :accessor seconds :initarg :seconds)))
#<STANDARD-CLASS MYTIME 422015CDD3>
CL-USER 24 > (let ((mt (make-instance 'mytime :seconds 100)))
(values (seconds mt)
(truncate (seconds mt) 60)))
100
1
通过访问器函数更短地访问插槽
Common Lisp 也有一种形式 WITH-ACCESSORS
。它允许我们在代码中使用符号而不是访问器形式 - 对于某个 CLOS 对象。在下面的示例中,我们可以使用 secs
,它在代码中看起来像一个变量,但 Common Lisp 会确保它实际上调用了访问器 seconds
。我们可以写 secs
而不是 (seconds mt)
。因此,它有助于使封闭的代码更短。将下一个示例与上面的代码进行比较。
CL-USER 25 > (let ((mt (make-instance 'mytime :seconds 200)))
(with-accessors ((secs seconds))
mt
(values secs (truncate secs 60))))
200
3
通过SLOT-VALUE
更短的插槽访问
CLOS 也有 WITH-SLOTS
用于通过插槽名称访问插槽,这里可以通过名称 secs
访问 mytime
实例的名为 seconds
的插槽:
CL-USER 26 > (let ((mt (make-instance 'mytime :seconds 200)))
(with-slots ((secs seconds))
mt
(values secs (truncate secs 60))))
200
3
如果您有一个 class 只是包装了一个对象,并且该对象具有已知类型,那么您始终可以只为该对象的 class 编写方法:
(defmethod time-as-unix-time ((tm integer))
(- tm (load-time-value (encode-universal-time 0 0 0 1 1 1970 0))))
例如
当然,如果面向对象的狂热者发现你在做这种事情,他们会把你扔进一个满是尖刺的坑里:这无疑违反了封装或其他一些邪教规则。
我想知道如何存储单个变量并在该变量上使用特定函数。我想知道是否有创建 class.
的替代方法具体来说,我正在创建一个应用程序,我在其中存储一个时间值,表示从基准时间(例如,2000 年 1 月 1 日,00:00:00)开始经过的秒数。我想对该值执行操作,例如将其从秒数转换为特定时间或日期,或从日期转换为特定秒数。
我已经使用 class 完成了此操作,但它看起来很浪费。具体来说,每次我访问经过的秒数的存储值时,它看起来类似于 (time-time time)
,其中 time-time
是时间实例 time
.
是否有更好的设计方法,也许没有 classes?
您可以像这样在词法作用域上尝试闭包:
(let ((time (get-universal-time)))
(defun set-time(tm)
(setf time tm))
(defun get-time()
time)
(defun get-time-in-some-other-format()
;; calculate return value based on 'time'
)
)
访问者名称
您可以在 CLOS 中以任何您喜欢的方式命名访问器。访问器函数可以称为 seconds
:
CL-USER 23 > (defclass mytime ()
((seconds :accessor seconds :initarg :seconds)))
#<STANDARD-CLASS MYTIME 422015CDD3>
CL-USER 24 > (let ((mt (make-instance 'mytime :seconds 100)))
(values (seconds mt)
(truncate (seconds mt) 60)))
100
1
通过访问器函数更短地访问插槽
Common Lisp 也有一种形式 WITH-ACCESSORS
。它允许我们在代码中使用符号而不是访问器形式 - 对于某个 CLOS 对象。在下面的示例中,我们可以使用 secs
,它在代码中看起来像一个变量,但 Common Lisp 会确保它实际上调用了访问器 seconds
。我们可以写 secs
而不是 (seconds mt)
。因此,它有助于使封闭的代码更短。将下一个示例与上面的代码进行比较。
CL-USER 25 > (let ((mt (make-instance 'mytime :seconds 200)))
(with-accessors ((secs seconds))
mt
(values secs (truncate secs 60))))
200
3
通过SLOT-VALUE
CLOS 也有 WITH-SLOTS
用于通过插槽名称访问插槽,这里可以通过名称 secs
访问 mytime
实例的名为 seconds
的插槽:
CL-USER 26 > (let ((mt (make-instance 'mytime :seconds 200)))
(with-slots ((secs seconds))
mt
(values secs (truncate secs 60))))
200
3
如果您有一个 class 只是包装了一个对象,并且该对象具有已知类型,那么您始终可以只为该对象的 class 编写方法:
(defmethod time-as-unix-time ((tm integer))
(- tm (load-time-value (encode-universal-time 0 0 0 1 1 1970 0))))
例如
当然,如果面向对象的狂热者发现你在做这种事情,他们会把你扔进一个满是尖刺的坑里:这无疑违反了封装或其他一些邪教规则。