Purpose: Functions, recursive and iterative: the sordid truth and other stuff
5.1 Let's talk recursive
Suppose we are (still) trying to write a function which takes two arguments - a list and a number - and if the number is = to the any element of the list then return the position of that element in the list. (So if the number matches the first element in the list return 0, if it matches the second return 1, and so on.) If the number isn't found in the list, we'll return nil.
Here's one solution to the problem:
(defun position-one (list number)And here's another:
(if list
(if (= (first list) number)
0
(let* ((pos (position-one (rest list) number)))
(if pos
(1+ pos)
nil)))))
5.2 Sinking without a trace
If you're stuck on seeing how these recursive functions work (and there
are plenty of further examples on the solution
sheets), you should consider tracing function calls. For example,
if you evaluate
(trace position-two)
- after defining the function! - and then call it with test data, you
will see a printout of all the recursive calls and their return values.
5.3 Divertimento: several functions, a macro and a special operator
1+
1- |
These functions each take one argument (a number) and return it after
either adding or subtracting 1, as the case may be.
(1+ foo) is shorthand for (+ foo 1) and similarly (1- foo) abbreviates (- foo 1) |
consp | Predicate, takes one argument and returns true if this is a cons, i.e.
a non-nil list.
There is another predicate, atom, which does the opposite, i.e. returns true if its argument is not a cons. |
listp | Predicate, takes one argument and returns true if this is a list, i.e. either a cons or nil. |
null | Predicate, true of empty lists. Now, given the existence of not, which is true if its argument is logically false, and given the equivalence between logical false and the empty list, you might be justified in asking: why bother? These two functions are the same. The answer is: use whichever looks more appropriate, and keep your code readable. Thank you. |
type-of | Takes one argument and returns its type. Neat. Play with this!
Note that the types returned by type-of will be the most specific the system can come up with, which is not always the most useful answer for you. For instance, (type-of "Hello World") => simple-base-string (as opposed to string). |
typep | Predicate for determining whether its first argument is of the type
named by its second argument.
(typep "Hello World" 'simple-base-string) => t Note that you do not have to be fully specific with the type here: (typep "Hello World" 'string) => t which is handy, because you can ask about the types you cared about without fussing at all about the underlying depths if that's what you wanted. So by of the type above, what I meant was either it is of this type or it is a subtype of this type. All common types are associated with a type predicate. We met
three of these above:
You lecturer will now draw a diagram showing how all these types fit together. |
equal | Predicate of two arguments: true if they are the "same". (Nebulous
concept, needs defining. At this stage in the game, if two objects print
out identically then they are equal. For what it's worth, if their
types are different then they cannot be equal.) For example:
(equal (list 'foo 'bar 'baz) '(foo bar baz)) => t (equal 'nick (car '(nick))) => t (equal 'nick "NICK") => nil |
nth | setfable Takes a number and a list and uses the number to index
into that list. (Zero based.) Hideously inefficient with long lists, compared
with equivalent operation on arrays. Similarly
nthcdr for the
nth cdr of the list.
(nth 0 '(1 2 3 4 5 6 7)) => 1 (equal any-list (nthcdr 0 any-list)) => t |
length | Takes a list and returns its length. As with nth, has to traverse
the whole list.
(length '(1 2 3 4 5 6 7)) => 7 (length nil) => 0 |
last | Returns the last cons in a list (the actual cons, not a copy of it):
(last '(1 2 3 4 5 6 7)) => (7) As above, takes longer with longer lists, but at least it's not as bad (why?) as (let* ((list '(1 2 3 4 5 6 7))) (nthcdr (1- (length list)) list)) Related to last is butlast, which returns a fresh copy of a list but with the final element missed off. Both last and butlast take an optional second argument, in case you want to include / exclude more than one element: (last '(1 2 3 4 5 6 7) 2) => (6 7) |
copy-list | Takes a list and returns a fresh copy of it. Each list is now safe
from destructive modification of the other.
(let* ((original-list (list 0 2 3 4)) (copied-list (copy-list original-list))) (incf (first original-list)) copied-list) => (0 2 3 4) An impressive number of supposedly intractable bugs in major applications can be traced back to failure to copy a list before modifying its guts. You have been warned. |
reverse | Takes a list and returns a fresh list with the old elements in reverse
order. Does not modify the old list.
(reverse '(1 2 3 4 5 6 7)) => (7 6 5 4 3 2 1) |
progn | Special operator. Evaluates all its subforms and returns the value
produced by the last subform.
(progn (call-this-function) (call-that-function) (1- 100)) => 99 Many other operators are said to contain implicit progns, for example defun. |
prog1 | Macro. Evaluates all its subforms and returns the value produced by the first subform. Useful for implementation of pop but not a lot else. |
5.4 Let's get iterative
Lisp supplies you with a whole load of macros and functions to make iteration simple. These range from fairly basic "loop forever until I tell you to stop" constructs to highly complex packages supporting specialized data structures, or lazy evaluation (see appendices to Cltl2). Let's start with the macro dolist:
(dolist (variable expression &optional result-form)This macro is used for iterating along a list. The expression is evaluated, and it should come out as a list. The variable variable is bound to the first element in the list and the forms form-1 ... form-n are evaluated as an implicit progn. Then variable is bound to the second element in the list and the forms are evaluated again, and so on down the list until it runs out.
form-1
...
form-n)
If result-form is present it is evaluated (with variable bound to nil, fwiw) and its value is the return value of dolist, otherwise dolist returns nil.
If you want to bail out of a dolist loop before the list has been used up, use the macro return. This takes an optional return value (to be evaluated), otherwise dolist still returns nil.
Examples:
(defun my-reverse (old-list)A similar macro is dotimes:
(let* ((new-list nil))
(dolist (elt old-list)
(push elt new-list))
new-list))
(my-reverse '(alpha beta gamma)) => (gamma beta alpha)(defun position-three (list number)
(let ((count 0))
(dolist (var list)
(if (= var number)
(return count))
(incf count))))
(dotimes (variable expression &optional result-form)This macro is used for iterating while incrementing a counter. The expression is evaluated, and this time it should come out as an integer. The variable variable is bound to 0 first time round the loop, then 1, etc until the loop has been been executed expression times. If result-form is present it is evaluated (with variable incremented one final time to give you the number of times the body of the loop was executed); return values and use of the macro return are exactly as for dolist. Examples:
form-1
...
form-n)
(dotimes (foo 10 foo)) => 10There is a fun example with palindromic strings on the HyperSpec page on dotimes.(setf bar 0)
(dotimes (foo 10)
(incf bar)) => nil
bar => 10(let* ((list nil))
(dotimes (i 5)
(push i list))
list) => (4 3 2 1 0)
5.5 Functions can iterate too
If you need to bomb down a list in a hurry (and it's your valuable time as a slothful typist I'm thinking of here), you should consider the following incredibly neat function.
(mapcar '1+ '(1 4 9 16)) => (2 5 10 17)So what does mapcar do? At its simplest it takes a function and a list. The function is passed members of the list, one at a time, and its return values are collected into a fresh list which is itself returned from mapcar.
(mapcar 'evenp '(1 2 3 4 5)) => (nil t nil t nil)
(mapcar 'type-of (list "my" (+ 1 2) 'sons))
=> (simple-base-string fixnum symbol)
(mapcar 'length '(() (()) (()()) (()()()) (()()()()))) => (0 1 2 3 4)
Hang on. Takes a function? How do you pass a function around? Answer: simple, just quote its name. As above.
Another example:
(defun my-copy-list (list)Stuck? Try tracing (see 5.2 above) the function which is being passed to mapcar, i.e. 1+, evenp, ..., identity in the above.
(mapcar 'identity list))
5.6 More on mapcar
In all the examples above, mapcar was passed a function which itself took one argument. Now, this need not be the case. More examples:
(mapcar '+ '(1 2 3) '(4 5 6)) => (5 7 9)The function handed to mapcar should take as many arguments as there are lists. First time around, it is called with the first element of each list, and so on. If the lists turn out to be of different lengths, mapcar halts as soon as any of them run out:
(mapcar '< '(1 2 3) '(3 2 1)) => (t nil nil)
(mapcar 'list '(one two three four) '(1 2 3 4) '("I" "II" "III" "IV"))
=> ((one 1 "I")
(two 2 "II")
(three 3 "III")
(four 4 "IV"))
(mapcar '/ '(0 1 2) '(3 2 1 0)) => (0 1/2 2) ; no division by zeroFinally, if you want to do iteration like this without the overhead of collecting results into a list, call mapc instead:
(mapc 'print (list foo bar baz))will print out the values of foo bar and baz. For want of anything better, mapc returns the first list passed to it, and programmers generally throw this value away. In other words, mapc is called for side effects only.
5.7 Practical session
MAKE SURE YOU UNDERSTAND IT ALL, NOW