Map function onto nested list?
Q: how can I map a function onto elements of nested lists?
For flat lists, we can use mapcar
to apply a function to each
element of the list:
(setq flat '("kittens" "puppies" "otters" "bunnies"))
(mapcar #'upcase flat) ; => ("KITTENS" "PUPPIES" "OTTERS" "BUNNIES")
But mapcar
doesn't work on nested lists:
(setq nested '(("kittens" "puppies")
("otters" "bunnies")))
(mapcar #'upcase nested) ; => ERROR
How can I map a function onto each element of a nested list, and
get back a new list with the same structure as the old one?
list mapping
add a comment |
Q: how can I map a function onto elements of nested lists?
For flat lists, we can use mapcar
to apply a function to each
element of the list:
(setq flat '("kittens" "puppies" "otters" "bunnies"))
(mapcar #'upcase flat) ; => ("KITTENS" "PUPPIES" "OTTERS" "BUNNIES")
But mapcar
doesn't work on nested lists:
(setq nested '(("kittens" "puppies")
("otters" "bunnies")))
(mapcar #'upcase nested) ; => ERROR
How can I map a function onto each element of a nested list, and
get back a new list with the same structure as the old one?
list mapping
2
One thing to note about recursive solutions using e.g.cl-labels
or-tree-map
is that Emacs doesn't handle recursion very efficiently and can easily blow themax-specpdl-size
stack. So, they may be fine for personal use, but Elisp libraries should generally strive for iterative solutions with an explicit stack.
– Basil
Dec 19 '18 at 13:24
@Basil: interesting point. Do you know of an iterative solution to avoid recursion?
– Dan♦
Dec 19 '18 at 13:42
add a comment |
Q: how can I map a function onto elements of nested lists?
For flat lists, we can use mapcar
to apply a function to each
element of the list:
(setq flat '("kittens" "puppies" "otters" "bunnies"))
(mapcar #'upcase flat) ; => ("KITTENS" "PUPPIES" "OTTERS" "BUNNIES")
But mapcar
doesn't work on nested lists:
(setq nested '(("kittens" "puppies")
("otters" "bunnies")))
(mapcar #'upcase nested) ; => ERROR
How can I map a function onto each element of a nested list, and
get back a new list with the same structure as the old one?
list mapping
Q: how can I map a function onto elements of nested lists?
For flat lists, we can use mapcar
to apply a function to each
element of the list:
(setq flat '("kittens" "puppies" "otters" "bunnies"))
(mapcar #'upcase flat) ; => ("KITTENS" "PUPPIES" "OTTERS" "BUNNIES")
But mapcar
doesn't work on nested lists:
(setq nested '(("kittens" "puppies")
("otters" "bunnies")))
(mapcar #'upcase nested) ; => ERROR
How can I map a function onto each element of a nested list, and
get back a new list with the same structure as the old one?
list mapping
list mapping
edited Dec 19 '18 at 18:51
Drew
46.9k462104
46.9k462104
asked Dec 19 '18 at 12:13
Dan♦
20.7k547107
20.7k547107
2
One thing to note about recursive solutions using e.g.cl-labels
or-tree-map
is that Emacs doesn't handle recursion very efficiently and can easily blow themax-specpdl-size
stack. So, they may be fine for personal use, but Elisp libraries should generally strive for iterative solutions with an explicit stack.
– Basil
Dec 19 '18 at 13:24
@Basil: interesting point. Do you know of an iterative solution to avoid recursion?
– Dan♦
Dec 19 '18 at 13:42
add a comment |
2
One thing to note about recursive solutions using e.g.cl-labels
or-tree-map
is that Emacs doesn't handle recursion very efficiently and can easily blow themax-specpdl-size
stack. So, they may be fine for personal use, but Elisp libraries should generally strive for iterative solutions with an explicit stack.
– Basil
Dec 19 '18 at 13:24
@Basil: interesting point. Do you know of an iterative solution to avoid recursion?
– Dan♦
Dec 19 '18 at 13:42
2
2
One thing to note about recursive solutions using e.g.
cl-labels
or -tree-map
is that Emacs doesn't handle recursion very efficiently and can easily blow the max-specpdl-size
stack. So, they may be fine for personal use, but Elisp libraries should generally strive for iterative solutions with an explicit stack.– Basil
Dec 19 '18 at 13:24
One thing to note about recursive solutions using e.g.
cl-labels
or -tree-map
is that Emacs doesn't handle recursion very efficiently and can easily blow the max-specpdl-size
stack. So, they may be fine for personal use, but Elisp libraries should generally strive for iterative solutions with an explicit stack.– Basil
Dec 19 '18 at 13:24
@Basil: interesting point. Do you know of an iterative solution to avoid recursion?
– Dan♦
Dec 19 '18 at 13:42
@Basil: interesting point. Do you know of an iterative solution to avoid recursion?
– Dan♦
Dec 19 '18 at 13:42
add a comment |
4 Answers
4
active
oldest
votes
You want to apply the solution recursively. For example:
(cl-labels ((upcase-nested
(x)
(cond ((consp x) (mapcar #'upcase-nested x))
((stringp x) (upcase x))
(t x))))
(mapcar #'upcase-nested
'(("kittens" ("spam") "puppies")
("otters" "bunnies" ("spam" ("spam" ("spam" ("spam" "and" "spam")))) "gnu"))))
=> (("KITTENS" ("SPAM") "PUPPIES")
("OTTERS" "BUNNIES" ("SPAM" ("SPAM" ("SPAM" ("SPAM" "AND" "SPAM")))) "GNU"))
Thank you! I benchmarked this solution and the-tree-map
answer that @Tobias posted. It turns out this solution runs about twice as quickly as-tree-map
.
– Dan♦
Dec 19 '18 at 13:40
add a comment |
The function -tree-map
from the dash package does that:
(-tree-map #'upcase '(("kittens" "puppies")
("otters" "bunnies")))
(("KITTENS" "PUPPIES") ("OTTERS" "BUNNIES"))
Great! Thank you. The advantage of this option is that it's a simple function call. The (minor) disadvantage is that it requires the use of a non-built in package. The bigger disadvantage is that it turns out that it's about twice as slow as the answer @phils posted, which is why I accepted that one over this one.
– Dan♦
Dec 19 '18 at 13:40
add a comment |
Following up Basils comment on the problems with recursive solutions I give here an iterative solution.
It exploits a breadth first search and therefore needs a queue. My first attempt with a depths-first search caused me some trouble with creating the structure and setting the values of the return tree.
The queue impl comes with the answer but you can also use other libraries like elib for that purpose.
The tree
argument of map-tree-iter
must be a list. Atoms and dotted lists are not allowed.
Note that I did not care about execution speed for this proof of concept implementation.
(defun make-queue (&rest elements)
"Return queue structure.
A queue is a cons.
Its `cdr' is the actual data list and its `car' contains the last link
of the data list for fast tail update.
If (cdr QUEUE) is nil so is (car QUEUE)."
(cons (last elements) elements))
(defun queue-push-back (queue &rest new-elements)
"Append NEW-ELEMENT to QUEUE."
(if (car queue)
(setcar queue
(last (setcdr (car queue) new-elements)))
(setcar queue (last (setcdr queue new-elements)))))
(defmacro queue-front (queue)
"Return first element of QUEUE.
The returned value is a place setable with setf."
`(cadr ,queue))
(defmacro queue-pop-front (queue)
"Pop the first element of QUEUE.
The return value is nil if there is no first element."
`(prog1
(pop (cdr ,queue))
(unless (cdr ,queue)
(setcar ,queue nil))))
(define-inline queue-not-empty-p (queue)
"Return non-nil if QUEUE is not empty."
(inline-letevals
(queue)
(inline-quote (car ,queue))))
(define-inline queue-to-list (queue)
"Convert QUEUE to a list.
Note that QUEUE and the returned list
share content and structure.
If you want to modify the structure of the returned list
you should consider (copy-list (queue-to-list QUEUE))."
(inline-letevals (queue)
(inline-quote (cdr ,queue))))
(defun queue-empty-p (queue)
"Return non-nil if QUEUE is empty."
(null (queue-not-empty-p queue)))
(defun map-tree-iter (fun tree)
"Map FUN over TREE."
(let* ((queue (make-queue tree))
(ret (list nil))
(queue-ret (make-queue ret)))
(while (queue-not-empty-p queue)
(let ((front (queue-pop-front queue))
(front-ret (queue-pop-front queue-ret)))
(while front
(cond
((consp (car front))
(queue-push-back queue (car front))
(let ((list-ret (list nil)))
(queue-push-back queue-ret list-ret)
(setcar front-ret list-ret)))
((car front)
(setcar front-ret (funcall fun (car front))))
)
(setq front (cdr front))
(when front
(setcdr front-ret (list nil))
(setq front-ret (cdr front-ret))))))
ret))
"Note that I did not care about execution speed" and yet you are usingdefine-inline
anddefmacro
. :)
– Basil
Dec 23 '18 at 14:03
add a comment |
Since you have a list of lists, you have to mapcar
over mapcar
:
(defun mapmap (f lol)
(mapcar (lambda (l) (mapcar f l)) lol))
(mapmap #'upcase nested)
;; => (("KITTENS" "PUPPIES") ("OTTERS" "BUNNIES"))
2
That only works for the rigid structure of a list of lists. The expression nested list actually stands for a tree structure. Your approach does not work for a more complicated tree.
– Tobias
Dec 19 '18 at 21:02
add a comment |
Your Answer
StackExchange.ready(function() {
var channelOptions = {
tags: "".split(" "),
id: "583"
};
initTagRenderer("".split(" "), "".split(" "), channelOptions);
StackExchange.using("externalEditor", function() {
// Have to fire editor after snippets, if snippets enabled
if (StackExchange.settings.snippets.snippetsEnabled) {
StackExchange.using("snippets", function() {
createEditor();
});
}
else {
createEditor();
}
});
function createEditor() {
StackExchange.prepareEditor({
heartbeatType: 'answer',
autoActivateHeartbeat: false,
convertImagesToLinks: false,
noModals: true,
showLowRepImageUploadWarning: true,
reputationToPostImages: null,
bindNavPrevention: true,
postfix: "",
imageUploader: {
brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
allowUrls: true
},
onDemand: true,
discardSelector: ".discard-answer"
,immediatelyShowMarkdownHelp:true
});
}
});
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
StackExchange.ready(
function () {
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2femacs.stackexchange.com%2fquestions%2f46680%2fmap-function-onto-nested-list%23new-answer', 'question_page');
}
);
Post as a guest
Required, but never shown
4 Answers
4
active
oldest
votes
4 Answers
4
active
oldest
votes
active
oldest
votes
active
oldest
votes
You want to apply the solution recursively. For example:
(cl-labels ((upcase-nested
(x)
(cond ((consp x) (mapcar #'upcase-nested x))
((stringp x) (upcase x))
(t x))))
(mapcar #'upcase-nested
'(("kittens" ("spam") "puppies")
("otters" "bunnies" ("spam" ("spam" ("spam" ("spam" "and" "spam")))) "gnu"))))
=> (("KITTENS" ("SPAM") "PUPPIES")
("OTTERS" "BUNNIES" ("SPAM" ("SPAM" ("SPAM" ("SPAM" "AND" "SPAM")))) "GNU"))
Thank you! I benchmarked this solution and the-tree-map
answer that @Tobias posted. It turns out this solution runs about twice as quickly as-tree-map
.
– Dan♦
Dec 19 '18 at 13:40
add a comment |
You want to apply the solution recursively. For example:
(cl-labels ((upcase-nested
(x)
(cond ((consp x) (mapcar #'upcase-nested x))
((stringp x) (upcase x))
(t x))))
(mapcar #'upcase-nested
'(("kittens" ("spam") "puppies")
("otters" "bunnies" ("spam" ("spam" ("spam" ("spam" "and" "spam")))) "gnu"))))
=> (("KITTENS" ("SPAM") "PUPPIES")
("OTTERS" "BUNNIES" ("SPAM" ("SPAM" ("SPAM" ("SPAM" "AND" "SPAM")))) "GNU"))
Thank you! I benchmarked this solution and the-tree-map
answer that @Tobias posted. It turns out this solution runs about twice as quickly as-tree-map
.
– Dan♦
Dec 19 '18 at 13:40
add a comment |
You want to apply the solution recursively. For example:
(cl-labels ((upcase-nested
(x)
(cond ((consp x) (mapcar #'upcase-nested x))
((stringp x) (upcase x))
(t x))))
(mapcar #'upcase-nested
'(("kittens" ("spam") "puppies")
("otters" "bunnies" ("spam" ("spam" ("spam" ("spam" "and" "spam")))) "gnu"))))
=> (("KITTENS" ("SPAM") "PUPPIES")
("OTTERS" "BUNNIES" ("SPAM" ("SPAM" ("SPAM" ("SPAM" "AND" "SPAM")))) "GNU"))
You want to apply the solution recursively. For example:
(cl-labels ((upcase-nested
(x)
(cond ((consp x) (mapcar #'upcase-nested x))
((stringp x) (upcase x))
(t x))))
(mapcar #'upcase-nested
'(("kittens" ("spam") "puppies")
("otters" "bunnies" ("spam" ("spam" ("spam" ("spam" "and" "spam")))) "gnu"))))
=> (("KITTENS" ("SPAM") "PUPPIES")
("OTTERS" "BUNNIES" ("SPAM" ("SPAM" ("SPAM" ("SPAM" "AND" "SPAM")))) "GNU"))
answered Dec 19 '18 at 12:40
phils
25.9k23566
25.9k23566
Thank you! I benchmarked this solution and the-tree-map
answer that @Tobias posted. It turns out this solution runs about twice as quickly as-tree-map
.
– Dan♦
Dec 19 '18 at 13:40
add a comment |
Thank you! I benchmarked this solution and the-tree-map
answer that @Tobias posted. It turns out this solution runs about twice as quickly as-tree-map
.
– Dan♦
Dec 19 '18 at 13:40
Thank you! I benchmarked this solution and the
-tree-map
answer that @Tobias posted. It turns out this solution runs about twice as quickly as -tree-map
.– Dan♦
Dec 19 '18 at 13:40
Thank you! I benchmarked this solution and the
-tree-map
answer that @Tobias posted. It turns out this solution runs about twice as quickly as -tree-map
.– Dan♦
Dec 19 '18 at 13:40
add a comment |
The function -tree-map
from the dash package does that:
(-tree-map #'upcase '(("kittens" "puppies")
("otters" "bunnies")))
(("KITTENS" "PUPPIES") ("OTTERS" "BUNNIES"))
Great! Thank you. The advantage of this option is that it's a simple function call. The (minor) disadvantage is that it requires the use of a non-built in package. The bigger disadvantage is that it turns out that it's about twice as slow as the answer @phils posted, which is why I accepted that one over this one.
– Dan♦
Dec 19 '18 at 13:40
add a comment |
The function -tree-map
from the dash package does that:
(-tree-map #'upcase '(("kittens" "puppies")
("otters" "bunnies")))
(("KITTENS" "PUPPIES") ("OTTERS" "BUNNIES"))
Great! Thank you. The advantage of this option is that it's a simple function call. The (minor) disadvantage is that it requires the use of a non-built in package. The bigger disadvantage is that it turns out that it's about twice as slow as the answer @phils posted, which is why I accepted that one over this one.
– Dan♦
Dec 19 '18 at 13:40
add a comment |
The function -tree-map
from the dash package does that:
(-tree-map #'upcase '(("kittens" "puppies")
("otters" "bunnies")))
(("KITTENS" "PUPPIES") ("OTTERS" "BUNNIES"))
The function -tree-map
from the dash package does that:
(-tree-map #'upcase '(("kittens" "puppies")
("otters" "bunnies")))
(("KITTENS" "PUPPIES") ("OTTERS" "BUNNIES"))
answered Dec 19 '18 at 12:42
Tobias
12.6k1833
12.6k1833
Great! Thank you. The advantage of this option is that it's a simple function call. The (minor) disadvantage is that it requires the use of a non-built in package. The bigger disadvantage is that it turns out that it's about twice as slow as the answer @phils posted, which is why I accepted that one over this one.
– Dan♦
Dec 19 '18 at 13:40
add a comment |
Great! Thank you. The advantage of this option is that it's a simple function call. The (minor) disadvantage is that it requires the use of a non-built in package. The bigger disadvantage is that it turns out that it's about twice as slow as the answer @phils posted, which is why I accepted that one over this one.
– Dan♦
Dec 19 '18 at 13:40
Great! Thank you. The advantage of this option is that it's a simple function call. The (minor) disadvantage is that it requires the use of a non-built in package. The bigger disadvantage is that it turns out that it's about twice as slow as the answer @phils posted, which is why I accepted that one over this one.
– Dan♦
Dec 19 '18 at 13:40
Great! Thank you. The advantage of this option is that it's a simple function call. The (minor) disadvantage is that it requires the use of a non-built in package. The bigger disadvantage is that it turns out that it's about twice as slow as the answer @phils posted, which is why I accepted that one over this one.
– Dan♦
Dec 19 '18 at 13:40
add a comment |
Following up Basils comment on the problems with recursive solutions I give here an iterative solution.
It exploits a breadth first search and therefore needs a queue. My first attempt with a depths-first search caused me some trouble with creating the structure and setting the values of the return tree.
The queue impl comes with the answer but you can also use other libraries like elib for that purpose.
The tree
argument of map-tree-iter
must be a list. Atoms and dotted lists are not allowed.
Note that I did not care about execution speed for this proof of concept implementation.
(defun make-queue (&rest elements)
"Return queue structure.
A queue is a cons.
Its `cdr' is the actual data list and its `car' contains the last link
of the data list for fast tail update.
If (cdr QUEUE) is nil so is (car QUEUE)."
(cons (last elements) elements))
(defun queue-push-back (queue &rest new-elements)
"Append NEW-ELEMENT to QUEUE."
(if (car queue)
(setcar queue
(last (setcdr (car queue) new-elements)))
(setcar queue (last (setcdr queue new-elements)))))
(defmacro queue-front (queue)
"Return first element of QUEUE.
The returned value is a place setable with setf."
`(cadr ,queue))
(defmacro queue-pop-front (queue)
"Pop the first element of QUEUE.
The return value is nil if there is no first element."
`(prog1
(pop (cdr ,queue))
(unless (cdr ,queue)
(setcar ,queue nil))))
(define-inline queue-not-empty-p (queue)
"Return non-nil if QUEUE is not empty."
(inline-letevals
(queue)
(inline-quote (car ,queue))))
(define-inline queue-to-list (queue)
"Convert QUEUE to a list.
Note that QUEUE and the returned list
share content and structure.
If you want to modify the structure of the returned list
you should consider (copy-list (queue-to-list QUEUE))."
(inline-letevals (queue)
(inline-quote (cdr ,queue))))
(defun queue-empty-p (queue)
"Return non-nil if QUEUE is empty."
(null (queue-not-empty-p queue)))
(defun map-tree-iter (fun tree)
"Map FUN over TREE."
(let* ((queue (make-queue tree))
(ret (list nil))
(queue-ret (make-queue ret)))
(while (queue-not-empty-p queue)
(let ((front (queue-pop-front queue))
(front-ret (queue-pop-front queue-ret)))
(while front
(cond
((consp (car front))
(queue-push-back queue (car front))
(let ((list-ret (list nil)))
(queue-push-back queue-ret list-ret)
(setcar front-ret list-ret)))
((car front)
(setcar front-ret (funcall fun (car front))))
)
(setq front (cdr front))
(when front
(setcdr front-ret (list nil))
(setq front-ret (cdr front-ret))))))
ret))
"Note that I did not care about execution speed" and yet you are usingdefine-inline
anddefmacro
. :)
– Basil
Dec 23 '18 at 14:03
add a comment |
Following up Basils comment on the problems with recursive solutions I give here an iterative solution.
It exploits a breadth first search and therefore needs a queue. My first attempt with a depths-first search caused me some trouble with creating the structure and setting the values of the return tree.
The queue impl comes with the answer but you can also use other libraries like elib for that purpose.
The tree
argument of map-tree-iter
must be a list. Atoms and dotted lists are not allowed.
Note that I did not care about execution speed for this proof of concept implementation.
(defun make-queue (&rest elements)
"Return queue structure.
A queue is a cons.
Its `cdr' is the actual data list and its `car' contains the last link
of the data list for fast tail update.
If (cdr QUEUE) is nil so is (car QUEUE)."
(cons (last elements) elements))
(defun queue-push-back (queue &rest new-elements)
"Append NEW-ELEMENT to QUEUE."
(if (car queue)
(setcar queue
(last (setcdr (car queue) new-elements)))
(setcar queue (last (setcdr queue new-elements)))))
(defmacro queue-front (queue)
"Return first element of QUEUE.
The returned value is a place setable with setf."
`(cadr ,queue))
(defmacro queue-pop-front (queue)
"Pop the first element of QUEUE.
The return value is nil if there is no first element."
`(prog1
(pop (cdr ,queue))
(unless (cdr ,queue)
(setcar ,queue nil))))
(define-inline queue-not-empty-p (queue)
"Return non-nil if QUEUE is not empty."
(inline-letevals
(queue)
(inline-quote (car ,queue))))
(define-inline queue-to-list (queue)
"Convert QUEUE to a list.
Note that QUEUE and the returned list
share content and structure.
If you want to modify the structure of the returned list
you should consider (copy-list (queue-to-list QUEUE))."
(inline-letevals (queue)
(inline-quote (cdr ,queue))))
(defun queue-empty-p (queue)
"Return non-nil if QUEUE is empty."
(null (queue-not-empty-p queue)))
(defun map-tree-iter (fun tree)
"Map FUN over TREE."
(let* ((queue (make-queue tree))
(ret (list nil))
(queue-ret (make-queue ret)))
(while (queue-not-empty-p queue)
(let ((front (queue-pop-front queue))
(front-ret (queue-pop-front queue-ret)))
(while front
(cond
((consp (car front))
(queue-push-back queue (car front))
(let ((list-ret (list nil)))
(queue-push-back queue-ret list-ret)
(setcar front-ret list-ret)))
((car front)
(setcar front-ret (funcall fun (car front))))
)
(setq front (cdr front))
(when front
(setcdr front-ret (list nil))
(setq front-ret (cdr front-ret))))))
ret))
"Note that I did not care about execution speed" and yet you are usingdefine-inline
anddefmacro
. :)
– Basil
Dec 23 '18 at 14:03
add a comment |
Following up Basils comment on the problems with recursive solutions I give here an iterative solution.
It exploits a breadth first search and therefore needs a queue. My first attempt with a depths-first search caused me some trouble with creating the structure and setting the values of the return tree.
The queue impl comes with the answer but you can also use other libraries like elib for that purpose.
The tree
argument of map-tree-iter
must be a list. Atoms and dotted lists are not allowed.
Note that I did not care about execution speed for this proof of concept implementation.
(defun make-queue (&rest elements)
"Return queue structure.
A queue is a cons.
Its `cdr' is the actual data list and its `car' contains the last link
of the data list for fast tail update.
If (cdr QUEUE) is nil so is (car QUEUE)."
(cons (last elements) elements))
(defun queue-push-back (queue &rest new-elements)
"Append NEW-ELEMENT to QUEUE."
(if (car queue)
(setcar queue
(last (setcdr (car queue) new-elements)))
(setcar queue (last (setcdr queue new-elements)))))
(defmacro queue-front (queue)
"Return first element of QUEUE.
The returned value is a place setable with setf."
`(cadr ,queue))
(defmacro queue-pop-front (queue)
"Pop the first element of QUEUE.
The return value is nil if there is no first element."
`(prog1
(pop (cdr ,queue))
(unless (cdr ,queue)
(setcar ,queue nil))))
(define-inline queue-not-empty-p (queue)
"Return non-nil if QUEUE is not empty."
(inline-letevals
(queue)
(inline-quote (car ,queue))))
(define-inline queue-to-list (queue)
"Convert QUEUE to a list.
Note that QUEUE and the returned list
share content and structure.
If you want to modify the structure of the returned list
you should consider (copy-list (queue-to-list QUEUE))."
(inline-letevals (queue)
(inline-quote (cdr ,queue))))
(defun queue-empty-p (queue)
"Return non-nil if QUEUE is empty."
(null (queue-not-empty-p queue)))
(defun map-tree-iter (fun tree)
"Map FUN over TREE."
(let* ((queue (make-queue tree))
(ret (list nil))
(queue-ret (make-queue ret)))
(while (queue-not-empty-p queue)
(let ((front (queue-pop-front queue))
(front-ret (queue-pop-front queue-ret)))
(while front
(cond
((consp (car front))
(queue-push-back queue (car front))
(let ((list-ret (list nil)))
(queue-push-back queue-ret list-ret)
(setcar front-ret list-ret)))
((car front)
(setcar front-ret (funcall fun (car front))))
)
(setq front (cdr front))
(when front
(setcdr front-ret (list nil))
(setq front-ret (cdr front-ret))))))
ret))
Following up Basils comment on the problems with recursive solutions I give here an iterative solution.
It exploits a breadth first search and therefore needs a queue. My first attempt with a depths-first search caused me some trouble with creating the structure and setting the values of the return tree.
The queue impl comes with the answer but you can also use other libraries like elib for that purpose.
The tree
argument of map-tree-iter
must be a list. Atoms and dotted lists are not allowed.
Note that I did not care about execution speed for this proof of concept implementation.
(defun make-queue (&rest elements)
"Return queue structure.
A queue is a cons.
Its `cdr' is the actual data list and its `car' contains the last link
of the data list for fast tail update.
If (cdr QUEUE) is nil so is (car QUEUE)."
(cons (last elements) elements))
(defun queue-push-back (queue &rest new-elements)
"Append NEW-ELEMENT to QUEUE."
(if (car queue)
(setcar queue
(last (setcdr (car queue) new-elements)))
(setcar queue (last (setcdr queue new-elements)))))
(defmacro queue-front (queue)
"Return first element of QUEUE.
The returned value is a place setable with setf."
`(cadr ,queue))
(defmacro queue-pop-front (queue)
"Pop the first element of QUEUE.
The return value is nil if there is no first element."
`(prog1
(pop (cdr ,queue))
(unless (cdr ,queue)
(setcar ,queue nil))))
(define-inline queue-not-empty-p (queue)
"Return non-nil if QUEUE is not empty."
(inline-letevals
(queue)
(inline-quote (car ,queue))))
(define-inline queue-to-list (queue)
"Convert QUEUE to a list.
Note that QUEUE and the returned list
share content and structure.
If you want to modify the structure of the returned list
you should consider (copy-list (queue-to-list QUEUE))."
(inline-letevals (queue)
(inline-quote (cdr ,queue))))
(defun queue-empty-p (queue)
"Return non-nil if QUEUE is empty."
(null (queue-not-empty-p queue)))
(defun map-tree-iter (fun tree)
"Map FUN over TREE."
(let* ((queue (make-queue tree))
(ret (list nil))
(queue-ret (make-queue ret)))
(while (queue-not-empty-p queue)
(let ((front (queue-pop-front queue))
(front-ret (queue-pop-front queue-ret)))
(while front
(cond
((consp (car front))
(queue-push-back queue (car front))
(let ((list-ret (list nil)))
(queue-push-back queue-ret list-ret)
(setcar front-ret list-ret)))
((car front)
(setcar front-ret (funcall fun (car front))))
)
(setq front (cdr front))
(when front
(setcdr front-ret (list nil))
(setq front-ret (cdr front-ret))))))
ret))
edited Dec 23 '18 at 13:18
answered Dec 23 '18 at 4:32
Tobias
12.6k1833
12.6k1833
"Note that I did not care about execution speed" and yet you are usingdefine-inline
anddefmacro
. :)
– Basil
Dec 23 '18 at 14:03
add a comment |
"Note that I did not care about execution speed" and yet you are usingdefine-inline
anddefmacro
. :)
– Basil
Dec 23 '18 at 14:03
"Note that I did not care about execution speed" and yet you are using
define-inline
and defmacro
. :)– Basil
Dec 23 '18 at 14:03
"Note that I did not care about execution speed" and yet you are using
define-inline
and defmacro
. :)– Basil
Dec 23 '18 at 14:03
add a comment |
Since you have a list of lists, you have to mapcar
over mapcar
:
(defun mapmap (f lol)
(mapcar (lambda (l) (mapcar f l)) lol))
(mapmap #'upcase nested)
;; => (("KITTENS" "PUPPIES") ("OTTERS" "BUNNIES"))
2
That only works for the rigid structure of a list of lists. The expression nested list actually stands for a tree structure. Your approach does not work for a more complicated tree.
– Tobias
Dec 19 '18 at 21:02
add a comment |
Since you have a list of lists, you have to mapcar
over mapcar
:
(defun mapmap (f lol)
(mapcar (lambda (l) (mapcar f l)) lol))
(mapmap #'upcase nested)
;; => (("KITTENS" "PUPPIES") ("OTTERS" "BUNNIES"))
2
That only works for the rigid structure of a list of lists. The expression nested list actually stands for a tree structure. Your approach does not work for a more complicated tree.
– Tobias
Dec 19 '18 at 21:02
add a comment |
Since you have a list of lists, you have to mapcar
over mapcar
:
(defun mapmap (f lol)
(mapcar (lambda (l) (mapcar f l)) lol))
(mapmap #'upcase nested)
;; => (("KITTENS" "PUPPIES") ("OTTERS" "BUNNIES"))
Since you have a list of lists, you have to mapcar
over mapcar
:
(defun mapmap (f lol)
(mapcar (lambda (l) (mapcar f l)) lol))
(mapmap #'upcase nested)
;; => (("KITTENS" "PUPPIES") ("OTTERS" "BUNNIES"))
answered Dec 19 '18 at 20:45
user3414663
285
285
2
That only works for the rigid structure of a list of lists. The expression nested list actually stands for a tree structure. Your approach does not work for a more complicated tree.
– Tobias
Dec 19 '18 at 21:02
add a comment |
2
That only works for the rigid structure of a list of lists. The expression nested list actually stands for a tree structure. Your approach does not work for a more complicated tree.
– Tobias
Dec 19 '18 at 21:02
2
2
That only works for the rigid structure of a list of lists. The expression nested list actually stands for a tree structure. Your approach does not work for a more complicated tree.
– Tobias
Dec 19 '18 at 21:02
That only works for the rigid structure of a list of lists. The expression nested list actually stands for a tree structure. Your approach does not work for a more complicated tree.
– Tobias
Dec 19 '18 at 21:02
add a comment |
Thanks for contributing an answer to Emacs Stack Exchange!
- Please be sure to answer the question. Provide details and share your research!
But avoid …
- Asking for help, clarification, or responding to other answers.
- Making statements based on opinion; back them up with references or personal experience.
To learn more, see our tips on writing great answers.
Some of your past answers have not been well-received, and you're in danger of being blocked from answering.
Please pay close attention to the following guidance:
- Please be sure to answer the question. Provide details and share your research!
But avoid …
- Asking for help, clarification, or responding to other answers.
- Making statements based on opinion; back them up with references or personal experience.
To learn more, see our tips on writing great answers.
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
StackExchange.ready(
function () {
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2femacs.stackexchange.com%2fquestions%2f46680%2fmap-function-onto-nested-list%23new-answer', 'question_page');
}
);
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
2
One thing to note about recursive solutions using e.g.
cl-labels
or-tree-map
is that Emacs doesn't handle recursion very efficiently and can easily blow themax-specpdl-size
stack. So, they may be fine for personal use, but Elisp libraries should generally strive for iterative solutions with an explicit stack.– Basil
Dec 19 '18 at 13:24
@Basil: interesting point. Do you know of an iterative solution to avoid recursion?
– Dan♦
Dec 19 '18 at 13:42