racket/collects/scribblings/guide/lambda.scrbl
Eli Barzilay debd1f9f1e Recketizing much in `net/*', mass ".ss" -> ".rkt" conversion in .scrbl files.
(Some other minor things here and there.)
2011-06-20 04:27:14 -04:00

245 lines
6.5 KiB
Racket

#lang scribble/doc
@(require scribble/manual scribble/eval "guide-utils.rkt")
@(define greet-eval (make-base-eval))
@title[#:tag "lambda"]{Functions@aux-elem{ (Procedures)}: @racket[lambda]}
A @racket[lambda] expression creates a function. In the simplest
case, a @racket[lambda] expression has the form
@specform[
(lambda (arg-id ...)
body ...+)
]
A @racket[lambda] form with @math{n} @racket[_arg-id]s accepts
@math{n} arguments:
@interaction[
((lambda (x) x)
1)
((lambda (x y) (+ x y))
1 2)
((lambda (x y) (+ x y))
1)
]
@;------------------------------------------------------------------------
@section[#:tag "rest-args"]{Declaring a Rest Argument}
A @racket[lambda] expression can also have the form
@specform[
(lambda rest-id
body ...+)
]
That is, a @racket[lambda] expression can have a single
@racket[_rest-id] that is not surrounded by parentheses. The resulting
function accepts any number of arguments, and the arguments are put
into a list bound to @racket[_rest-id].
@examples[
((lambda x x)
1 2 3)
((lambda x x))
((lambda x (car x))
1 2 3)
]
Functions with a @racket[_rest-id] often use @racket[apply] to call
another function that accepts any number of arguments.
@guideother{@secref["apply"] describes @racket[apply].}
@defexamples[
(define max-mag
(lambda nums
(apply max (map magnitude nums))))
(max 1 -2 0)
(max-mag 1 -2 0)
]
The @racket[lambda] form also supports required arguments combined
with a @racket[_rest-id]:
@specform[
(lambda (arg-id ...+ . rest-id)
body ...+)
]
The result of this form is a function that requires at least as many
arguments as @racket[_arg-id]s, and also accepts any number of
additional arguments.
@defexamples[
(define max-mag
(lambda (num . nums)
(apply max (map magnitude (cons num nums)))))
(max-mag 1 -2 0)
(max-mag)
]
A @racket[_rest-id] variable is sometimes called a @defterm{rest
argument}, because it accepts the ``rest'' of the function arguments.
@;------------------------------------------------------------------------
@section{Declaring Optional Arguments}
Instead of just an identifier, an argument (other than a rest
argument) in a @racket[lambda] form can be specified with an
identifier and a default value:
@specform/subs[
(lambda gen-formals
body ...+)
([gen-formals (arg ...)
rest-id
(arg ...+ . rest-id)]
[arg arg-id
[arg-id default-expr]])
]{}
An argument of the form @racket[[arg-id default-expr]] is
optional. When the argument is not supplied in an application,
@racket[_default-expr] produces the default value. The
@racket[_default-expr] can refer to any preceding @racket[_arg-id],
and every following @racket[_arg-id] must have a default as well.
@defexamples[
(define greet
(lambda (given [surname "Smith"])
(string-append "Hello, " given " " surname)))
(greet "John")
(greet "John" "Doe")
]
@def+int[
(define greet
(lambda (given [surname (if (equal? given "John")
"Doe"
"Smith")])
(string-append "Hello, " given " " surname)))
(greet "John")
(greet "Adam")
]
@section[#:tag "lambda-keywords"]{Declaring Keyword Arguments}
A @racket[lambda] form can declare an argument to be passed by
keyword, instead of position. Keyword arguments can be mixed with
by-position arguments, and default-value expressions can be supplied
for either kind of argument:
@guideother{@secref["keyword-args"] introduces function
calls with keywords.}
@specform/subs[
(lambda gen-formals
body ...+)
([gen-formals (arg ...)
rest-id
(arg ...+ . rest-id)]
[arg arg-id
[arg-id default-expr]
(code:line arg-keyword arg-id)
(code:line arg-keyword [arg-id default-expr])])
]{}
An argument specified as @racket[(code:line _arg-keyword _arg-id)] is
supplied by an application using the same @racket[_arg-keyword]. The
position of the keyword--identifier pair in the argument list does not
matter for matching with arguments in an application, because it will
be matched to an argument value by keyword instead of by position.
@def+int[
(define greet
(lambda (given #:last surname)
(string-append "Hello, " given " " surname)))
(greet "John" #:last "Smith")
(greet #:last "Doe" "John")
]
An @racket[(code:line _arg-keyword [_arg-id _default-expr])] argument
specifies a keyword-based argument with a default value.
@defexamples[
#:eval greet-eval
(define greet
(lambda (#:hi [hi "Hello"] given #:last [surname "Smith"])
(string-append hi ", " given " " surname)))
(greet "John")
(greet "Karl" #:last "Marx")
(greet "John" #:hi "Howdy")
(greet "Karl" #:last "Marx" #:hi "Guten Tag")
]
The @racket[lambda] form does not directly support the creation
of a function that accepts ``rest'' keywords. To construct a
function that accepts all keyword arguments, use
@racket[make-keyword-procedure]. The function supplied to
@racket[make-keyword-procedure] receives keyword arguments
through parallel lists in the first two (by-position) arguments,
and then all by-position arguments from an application as the
remaining by-position arguments.
@guideother{@secref["apply"] introduces @racket[keyword-apply].}
@defexamples[
#:eval greet-eval
(define (trace-wrap f)
(make-keyword-procedure
(lambda (kws kw-args . rest)
(printf "Called with ~s ~s ~s\n" kws kw-args rest)
(keyword-apply f kws kw-args rest))))
((trace-wrap greet) "John" #:hi "Howdy")
]
@refdetails["lambda"]{function expressions}
@;------------------------------------------------------------------------
@section[#:tag "case-lambda"]{Arity-Sensitive Functions: @racket[case-lambda]}
The @racket[case-lambda] form creates a function that can have
completely different behaviors depending on the number of arguments
that are supplied. A case-lambda expression has the form
@specform/subs[
(case-lambda
[formals body ...+]
...)
([formals (arg-id ...)
rest-id
(arg-id ...+ . rest-id)])
]
where each @racket[[_formals _body ...+]] is analogous to @racket[(lambda
_formals _body ...+)]. Applying a function produced by
@racket[case-lambda] is like applying a @racket[lambda] for the first
case that matches the number of given arguments.
@defexamples[
(define greet
(case-lambda
[(name) (string-append "Hello, " name)]
[(given surname) (string-append "Hello, " given " " surname)]))
(greet "John")
(greet "John" "Smith")
(greet)
]
A @racket[case-lambda] function cannot directly support optional or
keyword arguments.
@; ----------------------------------------------------------------------
@close-eval[greet-eval]