anaphoric/scribblings/anaphoric.scrbl
2021-07-28 19:54:00 +01:00

185 lines
7.0 KiB
Racket

#lang scribble/manual
@require[@for-label[anaphoric
racket/base]]
@title{Anaphoric macros}
@author[@author+email["Suzanne Soy" "racket@suzanne.soy"]]
@defmodule[anaphoric]
@section{Overview}
This package provides anaphoric versions of @racket[if],
@racket[when], @racket[cond], @racket[map], and @racket[filter]. These bind the syntax
parameter @racket[it] to the value produced by the
condition expression.
@racketblock[(aif (member 'a lst)
(displayln it)
(displayln "not found"))]
@racketblock[(awhen (member 'a lst)
(displayln it))]
@racketblock[(acond
[(member 'a lst) (displayln it)]
[(member 'b lst) (displayln it)]
[else (displayln "not found")])]
In the @racket[else] clause of @racket[acond] and in the
else branch of @racket[aif], the @racket[it] syntax
parameter keeps its value. This means it keeps the value
bound by the surrounding conditional, if any. Otherwise it
acts exactly as when it is used at the top-level, and raises
a syntax error.
@racketblock[(aif 'first
(aif (eq? 'second 'no)
'not-executed
(displayln it))
'not-executed)]
In the example above, @racket[(displayln it)] prints
@racket['first]. In the example below,
@racket[(displayln it)] raises a syntax error, as it appears
in a sequence of else branches:
@racketblock[(aif (eq? 'first 'no)
'not-executed
(aif (eq? 'second 'no)
'not-executed
(displayln it)))]
This package also provides the hygienic versions
@racket[if-let], @racket[when-let] and @racket[cond-let],
for which the user needs to specify an identifier instead of
using @racket[it].
@section{The anaphoric conditionals @racket[aif],
@racket[awhen] and @racket[acond]}
@defidform[it]{
Syntax parameter which acts as a rename transformer for
the result of the condition expression, when bound by
@racket[aif], @racket[awhen] or @racket[acond].
Raises a syntax error when used outside of the
@racket[_true-branch] of an @racket[aif] or the body of an
@racket[awhen] or the body of a non-@racket[else] case in
@racket[acond].}
@defform[(aif condition true-branch false-branch)]{
Variant of @racket[if] which binds @racket[it] to the
value of @racket[condition] in @racket[true-branch].
@racket[condition] is only evaluated once. In the
@racket[false-branch], @racket[it] is left unchanged.}
@defform[(awhen condition body ...+)]{
Variant of @racket[when] which binds @racket[it] to the
value of @racket[condition] in @racket[body ...+].
@racket[condition] is only evaluated once.}
@defform*[#:literals (else)
[(acond [conditionᵢ bodyᵢ ...+] ...)
(acond [conditionᵢ bodyᵢ ...+] ... [else body ...+])]]{
Variant of @racket[cond] which binds @racket[it] to the
corresponding @racket[conditionᵢ] in the non-@racket[else]
cases. More precisely, in each @racket[bodyᵢ ...+],
@racket[it] is bound to the value of the corresponding
@racket[conditionᵢ]. Each @racket[conditionᵢ] is evaluated
at most once (evaluation stops at the first successful
@racket[conditionᵢ]).}
@defform*[[(aand)
(aand conditionᵢ ... body)]]{
Variant of @racket[and] which binds @racket[it]
to the value of each @racket[conditionᵢ], in scope within the
next @racket[conditionᵢ] or @racket[body]. More precisely, the value
of each @racket[conditionᵢ] can be referred to as @racket[it] in
the following @racketvarfont{conditionᵢ₊₁}, and the value of the last
@racket[conditionᵢ] can be referred to as @racket[it] in the
@racket[body]. If there are no @racket[conditionᵢ], i.e. when
writing, @racket[(aand body)], then @racket[it] retains its original
binding (which means that @racket[it] could be unbound, e.g. if no
other @racketmodname[anaphoric] form wraps this one).
Each @racket[condition] is evaluated at most once, and
evaluation stops at the first false condition. The
@racket[body] is only evaluated when every
@racket[conditionᵢ] is successful.
}
@section{The hygienic versions @racket[if-let],
@racket[when-let] and @racket[cond-let]}
@defform[(if-let [identifier condition] true-branch false-branch)]{
Variant of @racket[if] which binds @racket[identifier] to
the value of @racket[condition] in @racket[true-branch].
@racket[condition] is only evaluated once. In the
@racket[false-branch], @racket[identifier] is left unchanged.}
@defform[(when-let [identifier condition] body ...+)]{
Variant of @racket[when] which binds @racket[identifier] to
the value of @racket[condition] in @racket[body ...+].
@racket[condition] is only evaluated once.}
@defform*[#:literals (else)
[(cond-let [[identifierᵢ conditionᵢ] bodyᵢ ...+] ...)
(cond-let [[identifierᵢ conditionᵢ] bodyᵢ ...+] ... [else body ...+])
(cond-let identifier [conditionᵢ bodyᵢ ...+] ...)
(cond-let identifier [conditionᵢ bodyᵢ ...+] ... [else body ...+])]]{
Variant of @racket[cond] which binds each
@racket[identifierᵢ] to the corresponding
@racket[conditionᵢ] in the non-@racket[else] cases. More
precisely, in each @racket[bodyᵢ ...+], the corresponding
@racket[identifierᵢ] is bound to the value of the
corresponding @racket[conditionᵢ].
The last two variants are shorthands for using the same
@racket[identifier] in all cases (except the @racket[else]
case).
Each @racket[conditionᵢ] is evaluated at most once
(evaluation stops at the first successful
@racket[conditionᵢ]).}
@defform*[[(and-let)
(and-let [identifier conditionᵢ] ... body)]]{
Variant of @racket[and] which binds each @racket[identifier]
to the value of its @racket[conditionᵢ], in scope within every
@racket[conditionᵢ] afterwards as well as in @racket[body].
Each @racket[conditionᵢ] is evaluated at most once, and
evaluation stops at the first false condition. The
@racket[body] is only evaluated when every
@racket[conditionᵢ] is successful.
}
@section{Anaphoric map and filter}
@defform[(amap body lst)]{
Anaphoric @racket[map]. Binds the syntax parameter @racket[it]
in the @racketid[body], and maps it over the list @racketid[lst]. Effectively the same
as wrapping the @racketid[body] in a @racket[lambda] with an @racket[it] parameter. Unlike @racket[map], @racket[amap]
only works on a single list.
@racket[amap] works with nested function calls:
@racketblock[(amap (string-append (string-upcase it) "!")
'("apple" "banana"))]
The syntax parameter @racket[it] may be used multiple times in the procedure:
@racketblock[(amap (* it it) '(1 2 3))]
}
@defform[(afilter body lst)]{
Anaphoric @racket[filter]. Binds the syntax parameter @racket[it]
in the @racketid[body], and filters the list @racketid[lst] using it. Effectively the same
as wrapping the body in a @racket[lambda] with an @racket[it] parameter.
@racket[afilter] works with nested function calls:
@racketblock[(afilter ((* it it) . > . 50) lst)]
}