Up: Conditional Execution [Contents][Index]
if expr clause {and clause}* [else clause {and clause}*] [end] when expr clause {and clause}* [else clause {and clause}*] [end] unless expr clause {and clause}* [else clause {and clause}*] [end]
The constructs ‘when’19 and ‘if’ allow conditional execution of loop clauses. These constructs are synonyms and can be used interchangeably.
expr
If the value of the test expression expr
is non-nil, the
expression clause1
is evaluated.
If the test expression evaluates to ‘nil’ and an ‘else’ construct is specified, the statements that follow the ‘else’ are evaluated; otherwise, control passes to the ‘next’ clause.
construct is equivalent to when (not expr)
and if (not expr
).20
If the value of the test expression expr
is nil, the expression
clause1 is evaluated.
If the test expression evaluates to non-nil and an ‘else’ construct is specified, the statements that follow the ‘else’ are evaluated; otherwise, control passes to the next clause.
The clause arguments must be either
Clauses that follow the test expression can be grouped by using the loop keyword ‘and’ to produce a compound clause.
keyword ‘it’ can be used to refer to the result of the test expression in a clause. If multiple clauses are connected with ‘and’, the ‘it’ construct must be used in the first clause in the block. Since ‘it’ is a loop keyword, it may not be used as a local variable within a loop.
If ‘when’ or ‘if’ clauses are nested, each ‘else’ is paired with the closest preceding ‘when’ or ‘if’ construct that has no associated else.
optional loop keyword marks the end of the clause. If this keyword is not specified, the next loop keyword marks the end. You can use ‘end’ to distinguish the scoping of compound clauses.
;;; Group conditional clauses into a block. (loop for i in numbers-list when (oddp i) do (print i) and collect i into odd-numbers and do (terpri) else ;I is even collect i into even-numbers finally (return (values odd-numbers even-numbers)))
;;; Collect numbers larger than 3. (loop for i in '(1 2 3 4 5 6) when (and (> i 3) i) collect it) ;it refers to (and (> i 3) i)
;;; Find a number in a list. (loop for i in '(1 2 3 4 5 6) when (and (> i 3) i) return it)
;; The preceding example is similar to the following one. (loop for i in '(1 2 3 4 5 6) thereis (and (> i 3) i))
;;; An example of using UNLESS with ELSE (yuk).-GLS (loop for turtle in teenage-mutant-ninja-turtles do (loop for x in '(joker brainiac shredder krazy-kat) unless (evil x) do (eat (make-pizza :anchovies t)) else unless (and (eq x 'shredder) (attacking-p x)) do (cut turtle slack);When the evil Shredder attacks, else (fight turtle x)));those turtle boys don't cut no slack
;;; Nest conditional clauses. (loop for i in list when (numberp i) when (bignump i) collect i into big-numbers else ;Not (bignump i) collect i into other-numbers else ;Not (numberp i) when (symbolp i) collect i into symbol-list else ;Not (symbolp i) (error "found a funny value in list ~S, value ~S~%" (list i)))
;;; Without the END marker, the last AND would apply to the ;;; inner IF rather than the outer one. (loop for x from 0 to 3 do (print x) if (zerop (mod x 2)) do (princ " a") and if (zerop (floor x 2)) do (princ " b") end and do (princ " c"))
[Compare this to the macro when, which does not allow an “else” part.-GLS]
[Compare this to the macro unless
, which does not allow an “else”
part—or do I mean a “then” part?! Ugh. To prevent confusion, I strongly
recommend as a matter of style that ‘else’ not be used with ‘unless’ loop
clauses.-GLS]
Up: Conditional Execution [Contents][Index]