Elisp: How to delete an element from an association list with string key

后端 未结 4 1286
执念已碎
执念已碎 2021-02-07 06:54

Now this works just fine:

(setq al \'((a . \"1\") (b . \"2\")))
(assq-delete-all \'a al)

But I\'m using strings as keys in my app:



        
相关标签:
4条回答
  • 2021-02-07 07:00

    If you know there can only be a single matching entry in your list, you can also use the following form:

    (setq al (delq (assoc <string> al) al)
    

    Notice that the setq (which was missing from your sample code) is very important for `delete' operations on lists, otherwise the operation fails when the deleted element happens to be the first on the list.

    0 讨论(0)
  • 2021-02-07 07:09

    The q in assq traditionally means eq equality is used for the objects.

    In other words, assq is an eq flavored assoc.

    Strings don't follow eq equality. Two strings which are equivalent character sequences might not be eq. The assoc in Emacs Lisp uses equal equality which works with strings.

    So what you need here is an assoc-delete-all for your equal-based association list, but that function doesn't exist.

    All I can find when I search for assoc-delete-all is this mailing list thread: http://lists.gnu.org/archive/html/emacs-devel/2005-07/msg00169.html

    Roll your own. It's fairly trivial: you march down the list, and collect all those entries into a new list whose car does not match the given key under equal.

    One useful thing to look at might be the Common Lisp compatibility library. http://www.gnu.org/software/emacs/manual/html_node/cl/index.html

    There are some useful functions there, like remove*, with which you can delete from a list with a custom predicate function for testing the elements. With that you can do something like this:

    ;; remove "a" from al, using equal as the test, applied to the car of each element
    (setq al (remove* "a" al :test 'equal :key 'car))
    

    The destructive variant is delete*.

    0 讨论(0)
  • 2021-02-07 07:12

    If using emacs 25 or newer you can use alist-get

    (setf (alist-get "a" al t t 'equal) t)

    0 讨论(0)
  • 2021-02-07 07:16

    Emacs 27+ includes assoc-delete-all which will work for string keys, and can also be used with arbitrary test functions.

    (assoc-delete-all KEY ALIST &optional TEST)
    
    Delete from ALIST all elements whose car is KEY.
    Compare keys with TEST.  Defaults to ‘equal’.
    Return the modified alist.
    Elements of ALIST that are not conses are ignored.
    

    e.g.:

    (setf ALIST (assoc-delete-all KEY ALIST))
    

    In earlier versions of Emacs, cl-delete provides an alternative:

    (setf ALIST (cl-delete KEY ALIST :key #'car :test #'equal))
    

    Which equivalently says to delete items from ALIST where the car of the list item is equal to KEY.

    n.b. The answer by Kaz mentions this latter option already, but using the older (require 'cl) names of delete* and remove*, whereas you would now (for supporting Emacs 24+) use cl-delete or cl-remove (which are auto-loaded).

    0 讨论(0)
提交回复
热议问题