doc-src/TutorialI/Misc/case_exprs.thy
author immler@in.tum.de
Thu, 26 Feb 2009 10:13:43 +0100
changeset 30151 629f3a92863e
parent 25340 6a3b20f0ae61
permissions -rw-r--r--
removed global ref dfg_format
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
9742
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
     1
(*<*)
16417
9bc16273c2d4 migrated theory headers to new format
haftmann
parents: 12699
diff changeset
     2
theory case_exprs imports Main begin
9742
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
     3
(*>*)
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
     4
25340
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
     5
text{*
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
     6
\subsection{Case Expressions}
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
     7
\label{sec:case-expressions}\index{*case expressions}%
11456
7eb63f63e6c6 revisions and indexing
paulson
parents: 11428
diff changeset
     8
HOL also features \isa{case}-expressions for analyzing
9742
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
     9
elements of a datatype. For example,
12699
deae80045527 *** empty log message ***
nipkow
parents: 11457
diff changeset
    10
@{term[display]"case xs of [] => [] | y#ys => y"}
deae80045527 *** empty log message ***
nipkow
parents: 11457
diff changeset
    11
evaluates to @{term"[]"} if @{term"xs"} is @{term"[]"} and to @{term"y"} if 
9742
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    12
@{term"xs"} is @{term"y#ys"}. (Since the result in both branches must be of
12699
deae80045527 *** empty log message ***
nipkow
parents: 11457
diff changeset
    13
the same type, it follows that @{term y} is of type @{typ"'a list"} and hence
deae80045527 *** empty log message ***
nipkow
parents: 11457
diff changeset
    14
that @{term xs} is of type @{typ"'a list list"}.)
9742
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    15
25340
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    16
In general, case expressions are of the form
9742
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    17
\[
25340
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    18
\begin{array}{c}
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    19
@{text"case"}~e~@{text"of"}\ pattern@1~@{text"\<Rightarrow>"}~e@1\ @{text"|"}\ \dots\
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    20
 @{text"|"}~pattern@m~@{text"\<Rightarrow>"}~e@m
9742
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    21
\end{array}
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    22
\]
25340
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    23
Like in functional programming, patterns are expressions consisting of
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    24
datatype constructors (e.g. @{term"[]"} and @{text"#"})
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    25
and variables, including the wildcard ``\verb$_$''.
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    26
Not all cases need to be covered and the order of cases matters.
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    27
However, one is well-advised not to wallow in complex patterns because
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    28
complex case distinctions tend to induce complex proofs.
9742
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    29
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    30
\begin{warn}
25340
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    31
Internally Isabelle only knows about exhaustive case expressions with
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    32
non-nested patterns: $pattern@i$ must be of the form
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    33
$C@i~x@ {i1}~\dots~x@ {ik@i}$ and $C@1, \dots, C@m$ must be exactly the
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    34
constructors of the type of $e$.
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    35
%
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    36
More complex case expressions are automatically
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    37
translated into the simpler form upon parsing but are not translated
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    38
back for printing. This may lead to surprising output.
9742
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    39
\end{warn}
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    40
25340
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    41
\begin{warn}
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    42
Like @{text"if"}, @{text"case"}-expressions may need to be enclosed in
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    43
parentheses to indicate their scope.
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    44
\end{warn}
9742
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    45
25340
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    46
\subsection{Structural Induction and Case Distinction}
6a3b20f0ae61 new general syntax
nipkow
parents: 16417
diff changeset
    47
\label{sec:struct-ind-case}
11456
7eb63f63e6c6 revisions and indexing
paulson
parents: 11428
diff changeset
    48
\index{case distinctions}\index{induction!structural}%
7eb63f63e6c6 revisions and indexing
paulson
parents: 11428
diff changeset
    49
Induction is invoked by \methdx{induct_tac}, as we have seen above; 
7eb63f63e6c6 revisions and indexing
paulson
parents: 11428
diff changeset
    50
it works for any datatype.  In some cases, induction is overkill and a case
7eb63f63e6c6 revisions and indexing
paulson
parents: 11428
diff changeset
    51
distinction over all constructors of the datatype suffices.  This is performed
7eb63f63e6c6 revisions and indexing
paulson
parents: 11428
diff changeset
    52
by \methdx{case_tac}.  Here is a trivial example:
9742
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    53
*}
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    54
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    55
lemma "(case xs of [] \<Rightarrow> [] | y#ys \<Rightarrow> xs) = xs";
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    56
apply(case_tac xs);
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    57
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    58
txt{*\noindent
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    59
results in the proof state
10420
ef006735bee8 *** empty log message ***
nipkow
parents: 10171
diff changeset
    60
@{subgoals[display,indent=0,margin=65]}
9742
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    61
which is solved automatically:
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    62
*}
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    63
10171
59d6633835fa *** empty log message ***
nipkow
parents: 9834
diff changeset
    64
apply(auto)
59d6633835fa *** empty log message ***
nipkow
parents: 9834
diff changeset
    65
(*<*)done(*>*)
9742
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    66
text{*
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    67
Note that we do not need to give a lemma a name if we do not intend to refer
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    68
to it explicitly in the future.
11456
7eb63f63e6c6 revisions and indexing
paulson
parents: 11428
diff changeset
    69
Other basic laws about a datatype are applied automatically during
7eb63f63e6c6 revisions and indexing
paulson
parents: 11428
diff changeset
    70
simplification, so no special methods are provided for them.
10824
4a212e635318 *** empty log message ***
nipkow
parents: 10420
diff changeset
    71
4a212e635318 *** empty log message ***
nipkow
parents: 10420
diff changeset
    72
\begin{warn}
4a212e635318 *** empty log message ***
nipkow
parents: 10420
diff changeset
    73
  Induction is only allowed on free (or \isasymAnd-bound) variables that
4a212e635318 *** empty log message ***
nipkow
parents: 10420
diff changeset
    74
  should not occur among the assumptions of the subgoal; see
4a212e635318 *** empty log message ***
nipkow
parents: 10420
diff changeset
    75
  \S\ref{sec:ind-var-in-prems} for details. Case distinction
4a212e635318 *** empty log message ***
nipkow
parents: 10420
diff changeset
    76
  (@{text"case_tac"}) works for arbitrary terms, which need to be
4a212e635318 *** empty log message ***
nipkow
parents: 10420
diff changeset
    77
  quoted if they are non-atomic. However, apart from @{text"\<And>"}-bound
4a212e635318 *** empty log message ***
nipkow
parents: 10420
diff changeset
    78
  variables, the terms must not contain variables that are bound outside.
4a212e635318 *** empty log message ***
nipkow
parents: 10420
diff changeset
    79
  For example, given the goal @{prop"\<forall>xs. xs = [] \<or> (\<exists>y ys. xs = y#ys)"},
4a212e635318 *** empty log message ***
nipkow
parents: 10420
diff changeset
    80
  @{text"case_tac xs"} will not work as expected because Isabelle interprets
4a212e635318 *** empty log message ***
nipkow
parents: 10420
diff changeset
    81
  the @{term xs} as a new free variable distinct from the bound
4a212e635318 *** empty log message ***
nipkow
parents: 10420
diff changeset
    82
  @{term xs} in the goal.
4a212e635318 *** empty log message ***
nipkow
parents: 10420
diff changeset
    83
\end{warn}
9742
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    84
*}
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    85
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    86
(*<*)
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    87
end
98d3ca2c18f7 *** empty log message ***
nipkow
parents:
diff changeset
    88
(*>*)