match.ss: Pattern Matching
To load: (require (lib "match.ss"))
This library provides functions for pattern-matching Scheme values. (This chapter adapted from Andrew K. Wright and Bruce Duba's original manual, entitled Pattern Matching for Scheme. The PLT Scheme port was originally done by Bruce Hauman and is maintained by Sam Tobin-Hochstadt.) The following forms are provided:
(match expr clause ...) (match-lambda clause ...) (match-lambda* clause ...) (match-let ((pat expr) ...) expr ···1) (match-let* ((pat expr) ...) expr ···1) (match-letrec ((pat expr) ...) expr ···1) (match-let var ((pat expr) ...) expr ···1) (match-define pat expr) clause is one of (pat expr ···1) (pat (=> identifier) expr ···1)
Figure 1 gives the full syntax for pat patterns.
| ||
| Figure 1: Pattern Syntax | ||
|
|
The next subsection describes the various patterns.
The match-lambda and match-lambda* forms are convenient
combinations of match and lambda, and can be explained
as follows:
where x is a unique variable. The
(match-lambda (pat expr ···1) ...)= (lambda (x) (match x (pat expr ···1) ...))(match-lambda* (pat expr ···1) ...)= (lambda x (match x (pat expr ···1) ...))
match-lambda form is convenient when defining a single argument
function that immediately destructures its argument.
The match-lambda* form constructs a function that accepts any number
of arguments; the patterns of match-lambda* should be lists.
The match-let, match-let*, match-letrec,
and match-define forms generalize
Scheme's let, let*, letrec, and define
expressions to allow
patterns in the binding position rather than just variables.
For example, the following expression:
(match-let ([(x y z) (list 1 2 3)]) body)
binds x to 1, y to 2, and z to 3 in the body.
These forms are convenient for destructuring the result
of a function that returns multiple values.
As usual for letrec and define,
pattern variables bound by match-letrec and match-define
should not be used in computing the bound value.
The match, match-lambda, and match-lambda* forms
allow the optional syntax (=> identifier) between the pattern
and the body of a clause. When the pattern match for such a clause
succeeds, the identifier is bound to a failure
procedure of zero arguments within the body. If this
procedure is invoked, it jumps back to the pattern matching
expression, and resumes the matching process as if the pattern had
failed to match. The body must not mutate the object being
matched, otherwise unpredictable behavior may result.
24.1 Patterns
Figure 1 gives the full syntax for patterns. Explanations of these patterns follow.
identifier(excluding the reserved names ?, =, $, _,and,or,not,set!,get!,..., and..kfor non-negative integersk) -- matches anything, and binds a variable of this name to the matching value in the body._-- matches anything, without binding any variables.#t,#f,string,number,character,'s-expression-- constant patterns that match themselves (i.e., the corresponding value must beto the pattern).equal?(matches a proper list of n elements that matchpat1···patn)pat1throughpatn.(generalizes the preceding pattern, where eachlvp1···lvpn)lvpcorresponds to a ``spliced'' list of greedy matches.For example,
(matches a proper list of n or more elements, where each element past the nth matchespat1···patnpatn+1...)patn+1. Each pattern variable inpatn+1is bound to a list of the matching values. For example, the expression:binds(match '(let ([x 1][y 2]) z) [('let ((binding vals) ...) exp) expr ···1])
bindingto the list'(x y),valsto the list'(1 2), andexpto'zin the body of thematch-expression. For the special case wherepatn+1is a pattern variable, the list bound to that variable may share with the matched value.Instead of
...or___(which are equivalent),..kor__kcan be used to match a sequence that is at leastklong. The pattern keywords..0,..., and___are equivalent.(-- matches a (possibly improper) list of at leastpat1···patn.patn+1)nelements that ends in something matchingpatn+1.(-- generalizes the preceding pattern with greedy-sequencelvp1···lvpn.patn+1)lvps.#(-- matches a vector of length n, whose elements match pat1 through patn. The generalization topat1···patn)lvps matches consecutive elements of the vector greedily.#&-- matches a box containing something matchingpatpat.(-- matches an instance of a structure type$struct-namepat1···patn)struct-name, where the instance containsnfields.Usually,
struct-nameis defined withdefine-struct. More generally,struct-namemust be bound to expansion-time information for a structure type (see section 12.6.4 in PLT MzScheme: Language Manual), where the information includes at least a predicate binding and some field accessor bindings (andpat1throughpatncorrespond to the provided accessors). In particular, a module import or aunit/sigimport with a signature containing astructdeclaration (see section 50.2) can provide the structure type information.(-- applies=fieldpat)fieldto the object being matched and usespatto match the extracted object. Thefieldsubexpression may be any expression, but is often useful as a struct selector.(-- matches if all of the subpatterns match. This pattern is often used asandpat1···patn)(to bindandxpat)xto the entire value that matchespat.(-- matches if any of the subpatterns match. At least one subpattern must be present. All subpatterns must bind the same set of pattern variables.orpat1···patn)(-- matches if none of the subpatterns match. The subpatterns may not bind any pattern variables.notpat1···patn)(-- In this pattern,?predicate-exprpat1···patn)predicate-exprmust be an expression evaluating to a single argument function. This pattern matches ifpredicate-exprapplied to the corresponding value is true, and the subpatternspat1throughpatnall match. Thepredicate-exprshould not have side effects, as the code generated by the pattern matcher may invoke predicates repeatedly in any order. Thepredicate-exprexpression is bound in the same scope as the match expression, so free variables inpredicate-exprare not bound by pattern variables.(-- matches anything, and bindsset!identifier)identifierto a procedure of one argument that mutates the corresponding field of the matching value. This pattern must be nested within a pair, vector, box, or structure pattern. For example, the expression:mutates the(define x (
list1 (list2 3))) (match x [(_ (_ (set! setit))) (setit 4)])cadadrof x to4, so thatxis'(1 (2 4)).(-- matches anything, and bindsget!identifier)identifierto a procedure of zero arguments that accesses the corresponding field of the matching value. This pattern is the complement toset!. As withset!, this pattern must be nested within a pair, vector, box, or structure pattern.`-- introduces a quasipattern, in which identifiers are considered to be symbolic constants. Like Scheme's quasiquote for data,quasipatternunquote(,) andunquote-splicing(,@) escape back to normal patterns.
If no clause matches the value, an exn:misc:match exception is raised.
24.2 Extending Match
There are two ways to extend or alter the behavior of match.
The parameter controls the behavior of
non-linear patterns:match-equality-test
(match-equality-test [expr]) PROCEDURE
When a variable appears more than once in a pattern, the values
matched by each instance are constrained to be the same in the sense
of the runtime value of . The default
value of this parameter is match-equality-test.equal?
The define-match-expander form extends the syntax of match
patterns:
( SYNTAX
define-match-expander id proc-expr)
( SYNTAX
define-match-expander id proc-expr proc-expr)
( SYNTAX
define-match-expander id proc-expr proc-expr proc-expr)
This form binds an identifier to a pattern transformer.
The first proc-expr subexpression must evaluate to a transformer
that produces a pattern in the syntax of Chapter 31.
Whenever id appears as the beginning of a pattern in a the
context of the pattern matching forms defined in Chapter 31, this
transformer is given, at expansion time, a syntax object
corresponding to the entire pattern (including id). The
pattern is the replaced with the result of the transformer.
If a second proc-expr subexpression is provided, it must produce
a similar transformer, but in the context of patterns written in the
syntax of the current chapter.
A transformer produced by a third proc-expr subexpression is
used when the id keyword is used in a traditional macro use
context. In this way, id can be given meaning both inside and
outside patterns.
24.3 Examples
This section illustrates the convenience of pattern matching with some examples. The following function recognizes some s-expressions that represent the standard Y operator:
(define Y? (match-lambda [('lambda (f1) ('lambda (y1) ((('lambda (x1) (f2 ('lambda (z1) ((x2 x3) z2)))) ('lambda (a1) (f3 ('lambda (b1) ((a2 a3) b2))))) y2))) (and (symbol?f1) (symbol?y1) (symbol?x1) (symbol?z1) (symbol?a1) (symbol?b1) (eq?f1 f2) (eq?f1 f3) (eq?y1 y2) (eq?x1 x2) (eq?x1 x3) (eq?z1 z2) (eq?a1 a2) (eq?a1 a3) (eq?b1 b2))] [_ #f]))
Writing an equivalent piece of code in raw Scheme is tedious.
The following code defines abstract syntax for a subset of Scheme, a parser into this abstract syntax, and an unparser.
(define-struct Lam (args body)) (define-struct Var (s)) (define-struct Const (n)) (define-struct App (fun args)) (define parse (match-lambda [(and s (?symbol?) (not 'lambda)) (make-Var s)] [(?number?n) (make-Const n)] [('lambda (and args ((?symbol?) ...) (not (? repeats?))) body) (make-Lam args (parse body))] [(f args ...) (make-App (parse f) (mapparse args))] [x (error'syntax "invalid expression")])) (define repeats? (lambda (l) (and (not(null?l)) (or (memq(carl) (cdrl)) (repeats? (cdrl)))))) (define unparse (match-lambda [($ Var s) s] [($ Const n) n] [($ Lam args body) `(lambda ,args ,(unparse body))] [($ App f args) `(,(unparse f) ,@(mapunparse args))]))
With pattern matching, it is easy to ensure that the parser rejects all incorrectly formed inputs with an error message.
With match-define, it is easy to define several procedures
that share a hidden variable. The following code defines three
procedures, inc, value, and reset,
that manipulate a hidden counter variable:
(match-define (inc value reset) (let ([val 0]) (list(lambda () (set! val (add1val))) (lambda () val) (lambda () (set! val 0)))))
Although this example is not recursive,
the bodies could recursively refer to each other.
The following code illustrates the creation of a match-expander that
works for both (lib "match.ss") and (lib
"plt-match.ss") syntax.
(require (prefix plt: (lib "plt-match.ss"))) (define-struct point (x y)) (define-match-expander Point (lambda (stx) (syntax-case stx () ((Point a b) #'(struct point (a b))))) (lambda (stx) (syntax-case stx () ((Point a b) #'($ point a b)))) (lambda (stx) (syntax-case stx () ((Point a b) #'(make-point a b))))) (define p (Point 3 4)) (match p ((Point x y) (+ x y))) ;; => 7 (plt:match p ((Point x y) (* x y))) ;; => 12