src/HOL/ex/Fundefs.thy
author blanchet
Tue, 10 Jun 2014 11:38:53 +0200
changeset 57199 472360558b22
parent 53991 d8f7f3d998de
child 58249 180f1b3508ed
permissions -rw-r--r--
tuning
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
     1
(*  Title:      HOL/ex/Fundefs.thy
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
     2
    Author:     Alexander Krauss, TU Muenchen
22726
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
     3
*)
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
     4
22726
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
     5
header {* Examples of function definitions *}
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
     6
19770
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
     7
theory Fundefs 
41413
64cd30d6b0b8 explicit file specifications -- avoid secondary load path;
wenzelm
parents: 40169
diff changeset
     8
imports Parity "~~/src/HOL/Library/Monad_Syntax"
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
     9
begin
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    10
22726
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
    11
subsection {* Very basic *}
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    12
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
    13
fun fib :: "nat \<Rightarrow> nat"
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
    14
where
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    15
  "fib 0 = 1"
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
    16
| "fib (Suc 0) = 1"
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
    17
| "fib (Suc (Suc n)) = fib n + fib (Suc n)"
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
    18
21319
cf814e36f788 replaced "auto_term" by the simpler method "relation", which does not try
krauss
parents: 21240
diff changeset
    19
text {* partial simp and induction rules: *}
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    20
thm fib.psimps
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
    21
thm fib.pinduct
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    22
19736
wenzelm
parents: 19583
diff changeset
    23
text {* There is also a cases rule to distinguish cases along the definition *}
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    24
thm fib.cases
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    25
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
    26
21319
cf814e36f788 replaced "auto_term" by the simpler method "relation", which does not try
krauss
parents: 21240
diff changeset
    27
text {* total simp and induction rules: *}
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    28
thm fib.simps
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    29
thm fib.induct
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    30
53611
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
    31
text {* elimination rules *}
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
    32
thm fib.elims
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
    33
22726
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
    34
subsection {* Currying *}
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    35
25170
krauss
parents: 24585
diff changeset
    36
fun add
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
    37
where
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    38
  "add 0 y = y"
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
    39
| "add (Suc x) y = Suc (add x y)"
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    40
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
    41
thm add.simps
19736
wenzelm
parents: 19583
diff changeset
    42
thm add.induct -- {* Note the curried induction predicate *}
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    43
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    44
22726
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
    45
subsection {* Nested recursion *}
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    46
25170
krauss
parents: 24585
diff changeset
    47
function nz 
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
    48
where
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    49
  "nz 0 = 0"
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
    50
| "nz (Suc x) = nz (nz x)"
21240
8e75fb38522c Made "termination by lexicographic_order" the default for "fun" definitions.
krauss
parents: 21051
diff changeset
    51
by pat_completeness auto
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
    52
19736
wenzelm
parents: 19583
diff changeset
    53
lemma nz_is_zero: -- {* A lemma we need to prove termination *}
21051
c49467a9c1e1 Switched function package to use the new package for inductive predicates.
krauss
parents: 20536
diff changeset
    54
  assumes trm: "nz_dom x"
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    55
  shows "nz x = 0"
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    56
using trm
39754
150f831ce4a3 no longer declare .psimps rules as [simp].
krauss
parents: 36270
diff changeset
    57
by induct (auto simp: nz.psimps)
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    58
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    59
termination nz
21319
cf814e36f788 replaced "auto_term" by the simpler method "relation", which does not try
krauss
parents: 21240
diff changeset
    60
  by (relation "less_than") (auto simp:nz_is_zero)
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    61
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    62
thm nz.simps
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    63
thm nz.induct
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    64
19770
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
    65
text {* Here comes McCarthy's 91-function *}
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
    66
21051
c49467a9c1e1 Switched function package to use the new package for inductive predicates.
krauss
parents: 20536
diff changeset
    67
21240
8e75fb38522c Made "termination by lexicographic_order" the default for "fun" definitions.
krauss
parents: 21051
diff changeset
    68
function f91 :: "nat => nat"
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
    69
where
19770
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
    70
  "f91 n = (if 100 < n then n - 10 else f91 (f91 (n + 11)))"
21240
8e75fb38522c Made "termination by lexicographic_order" the default for "fun" definitions.
krauss
parents: 21051
diff changeset
    71
by pat_completeness auto
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
    72
19770
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
    73
(* Prove a lemma before attempting a termination proof *)
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
    74
lemma f91_estimate: 
24585
haftmann
parents: 23817
diff changeset
    75
  assumes trm: "f91_dom n"
19770
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
    76
  shows "n < f91 n + 11"
39754
150f831ce4a3 no longer declare .psimps rules as [simp].
krauss
parents: 36270
diff changeset
    77
using trm by induct (auto simp: f91.psimps)
19770
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
    78
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
    79
termination
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
    80
proof
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
    81
  let ?R = "measure (%x. 101 - x)"
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
    82
  show "wf ?R" ..
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
    83
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
    84
  fix n::nat assume "~ 100 < n" (* Inner call *)
24585
haftmann
parents: 23817
diff changeset
    85
  thus "(n + 11, n) : ?R" by simp
19770
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
    86
21051
c49467a9c1e1 Switched function package to use the new package for inductive predicates.
krauss
parents: 20536
diff changeset
    87
  assume inner_trm: "f91_dom (n + 11)" (* Outer call *)
19770
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
    88
  with f91_estimate have "n + 11 < f91 (n + 11) + 11" .
20270
3abe7dae681e Function package can now do automatic splits of overlapping datatype patterns
krauss
parents: 19922
diff changeset
    89
  with `~ 100 < n` show "(f91 (n + 11), n) : ?R" by simp 
19770
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
    90
qed
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
    91
28584
58ac551ce1ce added lemma
nipkow
parents: 25680
diff changeset
    92
text{* Now trivial (even though it does not belong here): *}
58ac551ce1ce added lemma
nipkow
parents: 25680
diff changeset
    93
lemma "f91 n = (if 100 < n then n - 10 else 91)"
58ac551ce1ce added lemma
nipkow
parents: 25680
diff changeset
    94
by (induct n rule:f91.induct) auto
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    95
24585
haftmann
parents: 23817
diff changeset
    96
22726
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
    97
subsection {* More general patterns *}
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
    98
22726
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
    99
subsubsection {* Overlapping patterns *}
19782
48c4632e2c28 HOL/Tools/function_package: imporoved handling of guards, added an example
krauss
parents: 19770
diff changeset
   100
19736
wenzelm
parents: 19583
diff changeset
   101
text {* Currently, patterns must always be compatible with each other, since
20270
3abe7dae681e Function package can now do automatic splits of overlapping datatype patterns
krauss
parents: 19922
diff changeset
   102
no automatic splitting takes place. But the following definition of
19736
wenzelm
parents: 19583
diff changeset
   103
gcd is ok, although patterns overlap: *}
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   104
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
   105
fun gcd2 :: "nat \<Rightarrow> nat \<Rightarrow> nat"
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
   106
where
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   107
  "gcd2 x 0 = x"
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
   108
| "gcd2 0 y = y"
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
   109
| "gcd2 (Suc x) (Suc y) = (if x < y then gcd2 (Suc x) (y - x)
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
   110
                                    else gcd2 (x - y) (Suc y))"
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
   111
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   112
thm gcd2.simps
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   113
thm gcd2.induct
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   114
22726
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   115
subsubsection {* Guards *}
19782
48c4632e2c28 HOL/Tools/function_package: imporoved handling of guards, added an example
krauss
parents: 19770
diff changeset
   116
48c4632e2c28 HOL/Tools/function_package: imporoved handling of guards, added an example
krauss
parents: 19770
diff changeset
   117
text {* We can reformulate the above example using guarded patterns *}
48c4632e2c28 HOL/Tools/function_package: imporoved handling of guards, added an example
krauss
parents: 19770
diff changeset
   118
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
   119
function gcd3 :: "nat \<Rightarrow> nat \<Rightarrow> nat"
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
   120
where
19782
48c4632e2c28 HOL/Tools/function_package: imporoved handling of guards, added an example
krauss
parents: 19770
diff changeset
   121
  "gcd3 x 0 = x"
22492
43545e640877 Unified function syntax
krauss
parents: 22166
diff changeset
   122
| "gcd3 0 y = y"
43545e640877 Unified function syntax
krauss
parents: 22166
diff changeset
   123
| "x < y \<Longrightarrow> gcd3 (Suc x) (Suc y) = gcd3 (Suc x) (y - x)"
43545e640877 Unified function syntax
krauss
parents: 22166
diff changeset
   124
| "\<not> x < y \<Longrightarrow> gcd3 (Suc x) (Suc y) = gcd3 (x - y) (Suc y)"
19922
984ae977f7aa Fixed name clash.
krauss
parents: 19782
diff changeset
   125
  apply (case_tac x, case_tac a, auto)
984ae977f7aa Fixed name clash.
krauss
parents: 19782
diff changeset
   126
  apply (case_tac ba, auto)
19782
48c4632e2c28 HOL/Tools/function_package: imporoved handling of guards, added an example
krauss
parents: 19770
diff changeset
   127
  done
21240
8e75fb38522c Made "termination by lexicographic_order" the default for "fun" definitions.
krauss
parents: 21051
diff changeset
   128
termination by lexicographic_order
19782
48c4632e2c28 HOL/Tools/function_package: imporoved handling of guards, added an example
krauss
parents: 19770
diff changeset
   129
48c4632e2c28 HOL/Tools/function_package: imporoved handling of guards, added an example
krauss
parents: 19770
diff changeset
   130
thm gcd3.simps
48c4632e2c28 HOL/Tools/function_package: imporoved handling of guards, added an example
krauss
parents: 19770
diff changeset
   131
thm gcd3.induct
48c4632e2c28 HOL/Tools/function_package: imporoved handling of guards, added an example
krauss
parents: 19770
diff changeset
   132
48c4632e2c28 HOL/Tools/function_package: imporoved handling of guards, added an example
krauss
parents: 19770
diff changeset
   133
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
   134
text {* General patterns allow even strange definitions: *}
19782
48c4632e2c28 HOL/Tools/function_package: imporoved handling of guards, added an example
krauss
parents: 19770
diff changeset
   135
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
   136
function ev :: "nat \<Rightarrow> bool"
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
   137
where
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   138
  "ev (2 * n) = True"
22492
43545e640877 Unified function syntax
krauss
parents: 22166
diff changeset
   139
| "ev (2 * n + 1) = False"
19736
wenzelm
parents: 19583
diff changeset
   140
proof -  -- {* completeness is more difficult here \dots *}
19922
984ae977f7aa Fixed name clash.
krauss
parents: 19782
diff changeset
   141
  fix P :: bool
984ae977f7aa Fixed name clash.
krauss
parents: 19782
diff changeset
   142
    and x :: nat
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   143
  assume c1: "\<And>n. x = 2 * n \<Longrightarrow> P"
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   144
    and c2: "\<And>n. x = 2 * n + 1 \<Longrightarrow> P"
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   145
  have divmod: "x = 2 * (x div 2) + (x mod 2)" by auto
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   146
  show "P"
19736
wenzelm
parents: 19583
diff changeset
   147
  proof cases
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   148
    assume "x mod 2 = 0"
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   149
    with divmod have "x = 2 * (x div 2)" by simp
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   150
    with c1 show "P" .
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   151
  next
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   152
    assume "x mod 2 \<noteq> 0"
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   153
    hence "x mod 2 = 1" by simp
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   154
    with divmod have "x = 2 * (x div 2) + 1" by simp
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   155
    with c2 show "P" .
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   156
  qed
23315
df3a7e9ebadb tuned Proof
chaieb
parents: 23117
diff changeset
   157
qed presburger+ -- {* solve compatibility with presburger *} 
21240
8e75fb38522c Made "termination by lexicographic_order" the default for "fun" definitions.
krauss
parents: 21051
diff changeset
   158
termination by lexicographic_order
19568
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   159
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   160
thm ev.simps
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   161
thm ev.induct
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   162
thm ev.cases
6fa47aad35e9 Added small example theory for new function package.
krauss
parents:
diff changeset
   163
19770
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
   164
22726
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   165
subsection {* Mutual Recursion *}
19770
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
   166
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
   167
fun evn od :: "nat \<Rightarrow> bool"
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
   168
where
19770
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
   169
  "evn 0 = True"
20523
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
   170
| "od 0 = False"
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
   171
| "evn (Suc n) = od n"
36a59e5d0039 Major update to function package, including new syntax and the (only theoretical)
krauss
parents: 20270
diff changeset
   172
| "od (Suc n) = evn n"
19770
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
   173
21240
8e75fb38522c Made "termination by lexicographic_order" the default for "fun" definitions.
krauss
parents: 21051
diff changeset
   174
thm evn.simps
8e75fb38522c Made "termination by lexicographic_order" the default for "fun" definitions.
krauss
parents: 21051
diff changeset
   175
thm od.simps
19770
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
   176
23817
ee3ee9ea0d34 updated
krauss
parents: 23315
diff changeset
   177
thm evn_od.induct
19770
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
   178
thm evn_od.termination
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
   179
53611
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   180
thm evn.elims
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   181
thm od.elims
21240
8e75fb38522c Made "termination by lexicographic_order" the default for "fun" definitions.
krauss
parents: 21051
diff changeset
   182
22726
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   183
subsection {* Definitions in local contexts *}
22618
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   184
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   185
locale my_monoid = 
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   186
fixes opr :: "'a \<Rightarrow> 'a \<Rightarrow> 'a"
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   187
  and un :: "'a"
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   188
assumes assoc: "opr (opr x y) z = opr x (opr y z)"
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   189
  and lunit: "opr un x = x"
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   190
  and runit: "opr x un = x"
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   191
begin
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   192
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   193
fun foldR :: "'a list \<Rightarrow> 'a"
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   194
where
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   195
  "foldR [] = un"
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   196
| "foldR (x#xs) = opr x (foldR xs)"
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   197
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   198
fun foldL :: "'a list \<Rightarrow> 'a"
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   199
where
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   200
  "foldL [] = un"
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   201
| "foldL [x] = x"
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   202
| "foldL (x#y#ys) = foldL (opr x y # ys)" 
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   203
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   204
thm foldL.simps
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   205
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   206
lemma foldR_foldL: "foldR xs = foldL xs"
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   207
by (induct xs rule: foldL.induct) (auto simp:lunit runit assoc)
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   208
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   209
thm foldR_foldL
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   210
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   211
end
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   212
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   213
thm my_monoid.foldL.simps
e40957ccf0e9 added example for definitions in local contexts
krauss
parents: 22492
diff changeset
   214
thm my_monoid.foldR_foldL
19770
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
   215
53611
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   216
subsection {* @{text fun_cases} *}
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   217
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   218
subsubsection {* Predecessor *}
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   219
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   220
fun pred :: "nat \<Rightarrow> nat" where
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   221
"pred 0 = 0" |
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   222
"pred (Suc n) = n"
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   223
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   224
thm pred.elims
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   225
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   226
lemma assumes "pred x = y"
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   227
obtains "x = 0" "y = 0" | "n" where "x = Suc n" "y = n"
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   228
by (fact pred.elims[OF assms])
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   229
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   230
text {* If the predecessor of a number is 0, that number must be 0 or 1. *}
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   231
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   232
fun_cases pred0E[elim]: "pred n = 0"
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   233
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   234
lemma "pred n = 0 \<Longrightarrow> n = 0 \<or> n = Suc 0"
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   235
by (erule pred0E) metis+
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   236
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   237
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   238
text {* Other expressions on the right-hand side also work, but whether the
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   239
        generated rule is useful depends on how well the simplifier can
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   240
        simplify it. This example works well: *}
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   241
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   242
fun_cases pred42E[elim]: "pred n = 42"
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   243
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   244
lemma "pred n = 42 \<Longrightarrow> n = 43"
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   245
by (erule pred42E)
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   246
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   247
subsubsection {* List to option *}
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   248
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   249
fun list_to_option :: "'a list \<Rightarrow> 'a option" where
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   250
"list_to_option [x] = Some x" |
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   251
"list_to_option _ = None"
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   252
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   253
fun_cases list_to_option_NoneE: "list_to_option xs = None"
53991
d8f7f3d998de provide regular ML interface and use plain Syntax.read_prop/Syntax.check_prop (update by Manuel Eberl);
wenzelm
parents: 53611
diff changeset
   254
      and list_to_option_SomeE: "list_to_option xs = Some x"
53611
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   255
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   256
lemma "list_to_option xs = Some y \<Longrightarrow> xs = [y]"
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   257
by (erule list_to_option_SomeE)
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   258
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   259
subsubsection {* Boolean Functions *}
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   260
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   261
fun xor :: "bool \<Rightarrow> bool \<Rightarrow> bool" where
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   262
"xor False False = False" |
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   263
"xor True True = False" |
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   264
"xor _ _ = True"
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   265
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   266
thm xor.elims
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   267
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   268
text {* @{text fun_cases} does not only recognise function equations, but also works with
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   269
   functions that return a boolean, e.g.: *}
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   270
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   271
fun_cases xor_TrueE: "xor a b" and xor_FalseE: "\<not>xor a b"
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   272
print_theorems
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   273
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   274
subsubsection {* Many parameters *}
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   275
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   276
fun sum4 :: "nat \<Rightarrow> nat \<Rightarrow> nat \<Rightarrow> nat \<Rightarrow> nat" where
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   277
"sum4 a b c d = a + b + c + d"
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   278
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   279
fun_cases sum40E: "sum4 a b c d = 0"
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   280
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   281
lemma "sum4 a b c d = 0 \<Longrightarrow> a = 0"
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   282
by (erule sum40E)
437c0a63bb16 added some examples and tests for fun_cases
krauss
parents: 45008
diff changeset
   283
40111
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   284
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   285
subsection {* Partial Function Definitions *}
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   286
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   287
text {* Partial functions in the option monad: *}
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   288
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   289
partial_function (option)
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   290
  collatz :: "nat \<Rightarrow> nat list option"
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   291
where
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   292
  "collatz n =
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   293
  (if n \<le> 1 then Some [n]
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   294
   else if even n 
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   295
     then do { ns \<leftarrow> collatz (n div 2); Some (n # ns) }
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   296
     else do { ns \<leftarrow> collatz (3 * n + 1);  Some (n # ns)})"
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   297
40169
11ea439d947f declare recursive equation as ".simps", in accordance with other packages
krauss
parents: 40111
diff changeset
   298
declare collatz.simps[code]
40111
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   299
value "collatz 23"
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   300
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   301
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   302
text {* Tail-recursive functions: *}
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   303
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   304
partial_function (tailrec) fixpoint :: "('a \<Rightarrow> 'a) \<Rightarrow> 'a \<Rightarrow> 'a"
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   305
where
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   306
  "fixpoint f x = (if f x = x then x else fixpoint f (f x))"
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   307
80b7f456600f some partial_function examples
krauss
parents: 39754
diff changeset
   308
22726
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   309
subsection {* Regression tests *}
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   310
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   311
text {* The following examples mainly serve as tests for the 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   312
  function package *}
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   313
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   314
fun listlen :: "'a list \<Rightarrow> nat"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   315
where
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   316
  "listlen [] = 0"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   317
| "listlen (x#xs) = Suc (listlen xs)"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   318
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   319
(* Context recursion *)
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   320
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   321
fun f :: "nat \<Rightarrow> nat" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   322
where
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   323
  zero: "f 0 = 0"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   324
| succ: "f (Suc n) = (if f n = 0 then 0 else f n)"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   325
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   326
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   327
(* A combination of context and nested recursion *)
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   328
function h :: "nat \<Rightarrow> nat"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   329
where
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   330
  "h 0 = 0"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   331
| "h (Suc n) = (if h n = 0 then h (h n) else h n)"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   332
  by pat_completeness auto
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   333
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   334
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   335
(* Context, but no recursive call: *)
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   336
fun i :: "nat \<Rightarrow> nat"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   337
where
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   338
  "i 0 = 0"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   339
| "i (Suc n) = (if n = 0 then 0 else i n)"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   340
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   341
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   342
(* Tupled nested recursion *)
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   343
fun fa :: "nat \<Rightarrow> nat \<Rightarrow> nat"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   344
where
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   345
  "fa 0 y = 0"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   346
| "fa (Suc n) y = (if fa n y = 0 then 0 else fa n y)"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   347
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   348
(* Let *)
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   349
fun j :: "nat \<Rightarrow> nat"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   350
where
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   351
  "j 0 = 0"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   352
| "j (Suc n) = (let u = n  in Suc (j u))"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   353
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   354
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   355
(* There were some problems with fresh names\<dots> *)
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   356
function  k :: "nat \<Rightarrow> nat"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   357
where
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   358
  "k x = (let a = x; b = x in k x)"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   359
  by pat_completeness auto
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   360
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   361
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   362
function f2 :: "(nat \<times> nat) \<Rightarrow> (nat \<times> nat)"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   363
where
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   364
  "f2 p = (let (x,y) = p in f2 (y,x))"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   365
  by pat_completeness auto
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   366
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   367
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   368
(* abbreviations *)
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   369
fun f3 :: "'a set \<Rightarrow> bool"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   370
where
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   371
  "f3 x = finite x"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   372
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   373
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   374
(* Simple Higher-Order Recursion *)
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   375
datatype 'a tree = 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   376
  Leaf 'a 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   377
  | Branch "'a tree list"
23117
e2744f32641e updated examples to include an instance of (lexicographic_order simp:...)
krauss
parents: 22726
diff changeset
   378
36269
fa30cbb455df simplified example
krauss
parents: 28584
diff changeset
   379
fun treemap :: "('a \<Rightarrow> 'a) \<Rightarrow> 'a tree \<Rightarrow> 'a tree"
22726
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   380
where
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   381
  "treemap fn (Leaf n) = (Leaf (fn n))"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   382
| "treemap fn (Branch l) = (Branch (map (treemap fn) l))"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   383
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   384
fun tinc :: "nat tree \<Rightarrow> nat tree"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   385
where
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   386
  "tinc (Leaf n) = Leaf (Suc n)"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   387
| "tinc (Branch l) = Branch (map tinc l)"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   388
36270
fd95c0514623 tolerate eta-variants in f_graph.cases (from inductive package); added test case;
krauss
parents: 36269
diff changeset
   389
fun testcase :: "'a tree \<Rightarrow> 'a list"
fd95c0514623 tolerate eta-variants in f_graph.cases (from inductive package); added test case;
krauss
parents: 36269
diff changeset
   390
where
fd95c0514623 tolerate eta-variants in f_graph.cases (from inductive package); added test case;
krauss
parents: 36269
diff changeset
   391
  "testcase (Leaf a) = [a]"
fd95c0514623 tolerate eta-variants in f_graph.cases (from inductive package); added test case;
krauss
parents: 36269
diff changeset
   392
| "testcase (Branch x) =
fd95c0514623 tolerate eta-variants in f_graph.cases (from inductive package); added test case;
krauss
parents: 36269
diff changeset
   393
    (let xs = concat (map testcase x);
fd95c0514623 tolerate eta-variants in f_graph.cases (from inductive package); added test case;
krauss
parents: 36269
diff changeset
   394
         ys = concat (map testcase x) in
fd95c0514623 tolerate eta-variants in f_graph.cases (from inductive package); added test case;
krauss
parents: 36269
diff changeset
   395
     xs @ ys)"
fd95c0514623 tolerate eta-variants in f_graph.cases (from inductive package); added test case;
krauss
parents: 36269
diff changeset
   396
22726
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   397
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   398
(* Pattern matching on records *)
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   399
record point =
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   400
  Xcoord :: int
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   401
  Ycoord :: int
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   402
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   403
function swp :: "point \<Rightarrow> point"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   404
where
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   405
  "swp \<lparr> Xcoord = x, Ycoord = y \<rparr> = \<lparr> Xcoord = y, Ycoord = x \<rparr>"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   406
proof -
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   407
  fix P x
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   408
  assume "\<And>xa y. x = \<lparr>Xcoord = xa, Ycoord = y\<rparr> \<Longrightarrow> P"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   409
  thus "P"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   410
    by (cases x)
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   411
qed auto
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   412
termination by rule auto
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   413
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   414
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   415
(* The diagonal function *)
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   416
fun diag :: "bool \<Rightarrow> bool \<Rightarrow> bool \<Rightarrow> nat"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   417
where
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   418
  "diag x True False = 1"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   419
| "diag False y True = 2"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   420
| "diag True False z = 3"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   421
| "diag True True True = 4"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   422
| "diag False False False = 5"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   423
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   424
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   425
(* Many equations (quadratic blowup) *)
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   426
datatype DT = 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   427
  A | B | C | D | E | F | G | H | I | J | K | L | M | N | P
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   428
| Q | R | S | T | U | V
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   429
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   430
fun big :: "DT \<Rightarrow> nat"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   431
where
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   432
  "big A = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   433
| "big B = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   434
| "big C = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   435
| "big D = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   436
| "big E = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   437
| "big F = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   438
| "big G = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   439
| "big H = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   440
| "big I = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   441
| "big J = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   442
| "big K = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   443
| "big L = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   444
| "big M = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   445
| "big N = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   446
| "big P = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   447
| "big Q = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   448
| "big R = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   449
| "big S = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   450
| "big T = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   451
| "big U = 0" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   452
| "big V = 0"
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   453
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   454
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   455
(* automatic pattern splitting *)
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   456
fun
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   457
  f4 :: "nat \<Rightarrow> nat \<Rightarrow> bool" 
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   458
where
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   459
  "f4 0 0 = True"
25170
krauss
parents: 24585
diff changeset
   460
| "f4 _ _ = False"
22726
11e01dc78377 proper header, added regression tests
krauss
parents: 22618
diff changeset
   461
19770
be5c23ebe1eb HOL/Tools/function_package: Added support for mutual recursive definitions.
krauss
parents: 19736
diff changeset
   462
45008
8b74cfea913a match types when applying mono_thm -- previous export generalizes type variables;
krauss
parents: 41817
diff changeset
   463
(* polymorphic partial_function *)
8b74cfea913a match types when applying mono_thm -- previous export generalizes type variables;
krauss
parents: 41817
diff changeset
   464
partial_function (option) f5 :: "'a list \<Rightarrow> 'a option"
8b74cfea913a match types when applying mono_thm -- previous export generalizes type variables;
krauss
parents: 41817
diff changeset
   465
where
8b74cfea913a match types when applying mono_thm -- previous export generalizes type variables;
krauss
parents: 41817
diff changeset
   466
  "f5 x = f5 x"
8b74cfea913a match types when applying mono_thm -- previous export generalizes type variables;
krauss
parents: 41817
diff changeset
   467
19736
wenzelm
parents: 19583
diff changeset
   468
end