如何从 plist 中获取属性

emo*_*ull 2 lisp common-lisp string-to-symbol

我是 Lisp 的新手。我想使用这样的字符串变量从属性列表中访问特定属性

(setf sym (list :p1 1))
(setf x "p1")
(getf sym :x)
Run Code Online (Sandbox Code Playgroud)

Jos*_*lor 5

关于 cl:getf

让 Petit Prince 的回答是对的,getf可能是您想在这里使用的函数,但请注意,它不仅可以用于关键字符号。您可以将它用于任何对象。属性列表只是交替指标和值的列表,任何对象都可以是指标:

(let ((plist (list 'a 'b 'c 'd)))
  (getf plist 'c))
;=> D
Run Code Online (Sandbox Code Playgroud)

您甚至可以使用字符串作为指标:

(let* ((name "p1")
       (plist (list name 1)))
  (getf plist name))
;=> 1
Run Code Online (Sandbox Code Playgroud)

但是,这可能不是很好的做法,因为getf将指标与eq进行比较。这意味着使用字符串作为指标可能不可靠,具体取决于您的用例:

(let ((plist (list "p1" 1)))
  (getf plist "p1"))
;=> NIL
Run Code Online (Sandbox Code Playgroud)

对于你的例子

在您的情况下,您试图获取一个字符串并找到名称与字符串相等的符号的对象(即,具有相同的字符,但不考虑大小写)。循环遍历列表并将指标与string-equal进行比较可能更有意义。

(let ((plist '(:p1 1 :p2 2)))
  (loop 
     for (indicator value) on plist by #'cddr
     when (string-equal indicator "p1")
     return value))
;=> 1
Run Code Online (Sandbox Code Playgroud)

当然,您可以将其封装在抽象函数中:

(defun getf-string-equal (plist indicator)
  (loop
     for (i v) on plist by #'cddr
     when (string-equal i indicator)
     return v))
Run Code Online (Sandbox Code Playgroud)

(getf-string-equal '(:p1 1 :p2 2) "p1")
;=> 1
Run Code Online (Sandbox Code Playgroud)