src/HOL/Library/Poly_Mapping.thy
author nipkow
Tue, 17 Jun 2025 14:11:40 +0200
changeset 82733 8b537e1af2ec
parent 81816 bee084ecd18c
permissions -rw-r--r--
reinstated intersection of lists as inter_list_set
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
     1
(* Author: Andreas Lochbihler, ETH Zurich
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
     2
   Author: Florian Haftmann, TU Muenchen
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
     3
   with some material ported from HOL Light by LCP
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
     4
*)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
     5
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
     6
section \<open>Polynomial mapping: combination of almost everywhere zero functions with an algebraic view\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
     7
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
     8
theory Poly_Mapping
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
     9
imports Groups_Big_Fun Fun_Lexorder More_List
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    10
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    11
70045
7b6add80e3a5 fixed markup in Poly_Mapping; Free_Abelian_Groups (but not yet imported by Algebra!)
paulson <lp15@cam.ac.uk>
parents: 70043
diff changeset
    12
subsection \<open>Preliminary: auxiliary operations for \emph{almost everywhere zero}\<close>
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    13
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    14
text \<open>
70045
7b6add80e3a5 fixed markup in Poly_Mapping; Free_Abelian_Groups (but not yet imported by Algebra!)
paulson <lp15@cam.ac.uk>
parents: 70043
diff changeset
    15
  A central notion for polynomials are functions being \emph{almost everywhere zero}.
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    16
  For these we provide some auxiliary definitions and lemmas.
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    17
\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    18
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    19
lemma finite_mult_not_eq_zero_leftI:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    20
  fixes f :: "'b \<Rightarrow> 'a :: mult_zero"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    21
  assumes "finite {a. f a \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    22
  shows "finite {a. g a * f a \<noteq> 0}"
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
    23
  by (metis (mono_tags, lifting) Collect_mono assms mult_zero_right finite_subset)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    24
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    25
lemma finite_mult_not_eq_zero_rightI:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    26
  fixes f :: "'b \<Rightarrow> 'a :: mult_zero"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    27
  assumes "finite {a. f a \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    28
  shows "finite {a. f a * g a \<noteq> 0}"
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
    29
  by (metis (mono_tags, lifting) Collect_mono assms lambda_zero finite_subset)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    30
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    31
lemma finite_mult_not_eq_zero_prodI:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    32
  fixes f g :: "'a \<Rightarrow> 'b::semiring_0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    33
  assumes "finite {a. f a \<noteq> 0}" (is "finite ?F")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    34
  assumes "finite {b. g b \<noteq> 0}" (is "finite ?G")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    35
  shows "finite {(a, b). f a * g b \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    36
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    37
  from assms have "finite (?F \<times> ?G)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    38
    by blast
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    39
  then have "finite {(a, b). f a \<noteq> 0 \<and> g b \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    40
    by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    41
  then show ?thesis
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    42
    by (rule rev_finite_subset) auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    43
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    44
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    45
lemma finite_not_eq_zero_sumI:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    46
  fixes f g :: "'a::monoid_add \<Rightarrow> 'b::semiring_0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    47
  assumes "finite {a. f a \<noteq> 0}" (is "finite ?F")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    48
  assumes "finite {b. g b \<noteq> 0}" (is "finite ?G")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    49
  shows "finite {a + b | a b. f a \<noteq> 0 \<and> g b \<noteq> 0}" (is "finite ?FG")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    50
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    51
  from assms have "finite (?F \<times> ?G)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    52
    by (simp add: finite_cartesian_product_iff)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    53
  then have "finite (case_prod plus ` (?F \<times> ?G))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    54
    by (rule finite_imageI)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    55
  also have "case_prod plus ` (?F \<times> ?G) = ?FG"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    56
    by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    57
  finally show ?thesis
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    58
    by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    59
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    60
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    61
lemma finite_mult_not_eq_zero_sumI:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    62
  fixes f g :: "'a::monoid_add \<Rightarrow> 'b::semiring_0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    63
  assumes "finite {a. f a \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    64
  assumes "finite {b. g b \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    65
  shows "finite {a + b | a b. f a * g b \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    66
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    67
  from assms
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    68
  have "finite {a + b | a b. f a \<noteq> 0 \<and> g b \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    69
    by (rule finite_not_eq_zero_sumI)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    70
  then show ?thesis
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    71
    by (rule rev_finite_subset) (auto dest: mult_not_zero)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    72
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    73
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    74
lemma finite_Sum_any_not_eq_zero_weakenI:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    75
  assumes "finite {a. \<exists>b. f a b \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    76
  shows "finite {a. Sum_any (f a) \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    77
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    78
  have "{a. Sum_any (f a) \<noteq> 0} \<subseteq> {a. \<exists>b. f a b \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    79
    by (auto elim: Sum_any.not_neutral_obtains_not_neutral)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    80
  then show ?thesis using assms by (rule finite_subset)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    81
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    82
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    83
context zero
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    84
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    85
80914
d97fdabd9e2b standardize mixfix annotations via "isabelle update -a -u mixfix_cartouches" --- to simplify systematic editing;
wenzelm
parents: 80095
diff changeset
    86
definition "when" :: "'a \<Rightarrow> bool \<Rightarrow> 'a" (infixl \<open>when\<close> 20)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    87
where
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    88
  "(a when P) = (if P then a else 0)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    89
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    90
text \<open>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    91
  Case distinctions always complicate matters, particularly when
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    92
  nested.  The @{const "when"} operation allows to minimise these
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    93
  if @{term 0} is the false-case value and makes proof obligations
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    94
  much more readable.
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    95
\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    96
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    97
lemma "when" [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    98
  "P \<Longrightarrow> (a when P) = a"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
    99
  "\<not> P \<Longrightarrow> (a when P) = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   100
  by (simp_all add: when_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   101
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   102
lemma when_simps [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   103
  "(a when True) = a"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   104
  "(a when False) = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   105
  by simp_all
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   106
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   107
lemma when_cong:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   108
  assumes "P \<longleftrightarrow> Q"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   109
    and "Q \<Longrightarrow> a = b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   110
  shows "(a when P) = (b when Q)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   111
  using assms by (simp add: when_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   112
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   113
lemma zero_when [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   114
  "(0 when P) = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   115
  by (simp add: when_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   116
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   117
lemma when_when:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   118
  "(a when P when Q) = (a when P \<and> Q)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   119
  by (cases Q) simp_all
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   120
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   121
lemma when_commute:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   122
  "(a when Q when P) = (a when P when Q)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   123
  by (simp add: when_when conj_commute)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   124
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   125
lemma when_neq_zero [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   126
  "(a when P) \<noteq> 0 \<longleftrightarrow> P \<and> a \<noteq> 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   127
  by (cases P) simp_all
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   128
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   129
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   130
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   131
context monoid_add
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   132
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   133
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   134
lemma when_add_distrib:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   135
  "(a + b when P) = (a when P) + (b when P)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   136
  by (simp add: when_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   137
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   138
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   139
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   140
context semiring_1
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   141
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   142
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   143
lemma zero_power_eq:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   144
  "0 ^ n = (1 when n = 0)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   145
  by (simp add: power_0_left)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   146
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   147
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   148
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   149
context comm_monoid_add
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   150
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   151
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   152
lemma Sum_any_when_equal [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   153
  "(\<Sum>a. (f a when a = b)) = f b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   154
  by (simp add: when_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   155
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   156
lemma Sum_any_when_equal' [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   157
  "(\<Sum>a. (f a when b = a)) = f b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   158
  by (simp add: when_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   159
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   160
lemma Sum_any_when_independent:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   161
  "(\<Sum>a. g a when P) = ((\<Sum>a. g a) when P)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   162
  by (cases P) simp_all
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   163
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   164
lemma Sum_any_when_dependent_prod_right:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   165
  "(\<Sum>(a, b). g a when b = h a) = (\<Sum>a. g a)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   166
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   167
  have "inj_on (\<lambda>a. (a, h a)) {a. g a \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   168
    by (rule inj_onI) auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   169
  then show ?thesis unfolding Sum_any.expand_set
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   170
    by (rule sum.reindex_cong) auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   171
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   172
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   173
lemma Sum_any_when_dependent_prod_left:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   174
  "(\<Sum>(a, b). g b when a = h b) = (\<Sum>b. g b)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   175
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   176
  have "(\<Sum>(a, b). g b when a = h b) = (\<Sum>(b, a). g b when a = h b)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   177
    by (rule Sum_any.reindex_cong [of prod.swap]) (simp_all add: fun_eq_iff)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   178
  then show ?thesis by (simp add: Sum_any_when_dependent_prod_right)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   179
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   180
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   181
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   182
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   183
context cancel_comm_monoid_add
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   184
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   185
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   186
lemma when_diff_distrib:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   187
  "(a - b when P) = (a when P) - (b when P)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   188
  by (simp add: when_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   189
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   190
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   191
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   192
context group_add
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   193
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   194
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   195
lemma when_uminus_distrib:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   196
  "(- a when P) = - (a when P)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   197
  by (simp add: when_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   198
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   199
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   200
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   201
context mult_zero
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   202
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   203
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   204
lemma mult_when:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   205
  "a * (b when P) = (a * b when P)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   206
  by (cases P) simp_all
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   207
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   208
lemma when_mult:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   209
  "(a when P) * b = (a * b when P)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   210
  by (cases P) simp_all
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   211
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   212
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   213
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   214
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   215
subsection \<open>Type definition\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   216
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   217
text \<open>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   218
  The following type is of central importance:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   219
\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   220
80914
d97fdabd9e2b standardize mixfix annotations via "isabelle update -a -u mixfix_cartouches" --- to simplify systematic editing;
wenzelm
parents: 80095
diff changeset
   221
typedef (overloaded) ('a, 'b) poly_mapping (\<open>(_ \<Rightarrow>\<^sub>0 /_)\<close> [1, 0] 0) =
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   222
  "{f :: 'a \<Rightarrow> 'b::zero. finite {x. f x \<noteq> 0}}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   223
  morphisms lookup Abs_poly_mapping
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
   224
  using not_finite_existsD by force
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   225
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   226
declare lookup_inverse [simp]
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   227
declare lookup_inject [simp]
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   228
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   229
lemma lookup_Abs_poly_mapping [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   230
  "finite {x. f x \<noteq> 0} \<Longrightarrow> lookup (Abs_poly_mapping f) = f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   231
  using Abs_poly_mapping_inverse [of f] by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   232
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   233
lemma finite_lookup [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   234
  "finite {k. lookup f k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   235
  using poly_mapping.lookup [of f] by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   236
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   237
lemma finite_lookup_nat [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   238
  fixes f :: "'a \<Rightarrow>\<^sub>0 nat"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   239
  shows "finite {k. 0 < lookup f k}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   240
  using poly_mapping.lookup [of f] by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   241
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   242
lemma poly_mapping_eqI:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   243
  assumes "\<And>k. lookup f k = lookup g k"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   244
  shows "f = g"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   245
  using assms unfolding poly_mapping.lookup_inject [symmetric]
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   246
  by blast
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   247
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   248
lemma poly_mapping_eq_iff: "a = b \<longleftrightarrow> lookup a = lookup b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   249
  by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   250
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   251
text \<open>
70045
7b6add80e3a5 fixed markup in Poly_Mapping; Free_Abelian_Groups (but not yet imported by Algebra!)
paulson <lp15@cam.ac.uk>
parents: 70043
diff changeset
   252
  We model the universe of functions being \emph{almost everywhere zero}
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   253
  by means of a separate type @{typ "('a, 'b) poly_mapping"}.
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   254
  For convenience we provide a suggestive infix syntax which
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   255
  is a variant of the usual function space syntax.  Conversion between both types
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   256
  happens through the morphisms
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   257
  \begin{quote}
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   258
    @{term_type lookup}
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   259
  \end{quote}
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   260
  \begin{quote}
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   261
    @{term_type Abs_poly_mapping}
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   262
  \end{quote}
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   263
  satisfying
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   264
  \begin{quote}
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   265
    @{thm lookup_inverse}
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   266
  \end{quote}
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   267
  \begin{quote}
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   268
    @{thm lookup_Abs_poly_mapping}
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   269
  \end{quote}
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   270
  Luckily, we have rarely to deal with those low-level morphisms explicitly
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   271
  but rely on Isabelle's \emph{lifting} package with its method \<open>transfer\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   272
  and its specification tool \<open>lift_definition\<close>.
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   273
\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   274
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   275
setup_lifting type_definition_poly_mapping
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   276
code_datatype Abs_poly_mapping\<comment>\<open>FIXME? workaround for preventing \<open>code_abstype\<close> setup\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   277
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   278
text \<open>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   279
  @{typ "'a \<Rightarrow>\<^sub>0 'b"} serves distinctive purposes:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   280
  \begin{enumerate}
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   281
    \item A clever nesting as @{typ "(nat \<Rightarrow>\<^sub>0 nat) \<Rightarrow>\<^sub>0 'a"}
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   282
      later in theory \<open>MPoly\<close> gives a suitable
70045
7b6add80e3a5 fixed markup in Poly_Mapping; Free_Abelian_Groups (but not yet imported by Algebra!)
paulson <lp15@cam.ac.uk>
parents: 70043
diff changeset
   283
      representation type for polynomials \emph{almost for free}:
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   284
      Interpreting @{typ "nat \<Rightarrow>\<^sub>0 nat"} as a mapping from variable identifiers
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   285
      to exponents yields monomials, and the whole type maps monomials
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   286
      to coefficients.  Lets call this the \emph{ultimate interpretation}.
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   287
    \item A further more specialised type isomorphic to @{typ "nat \<Rightarrow>\<^sub>0 'a"}
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   288
      is apt to direct implementation using code generation
76987
4c275405faae isabelle update -u cite;
wenzelm
parents: 76484
diff changeset
   289
      \<^cite>\<open>"Haftmann-Nipkow:2010:code"\<close>.
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   290
  \end{enumerate}
70045
7b6add80e3a5 fixed markup in Poly_Mapping; Free_Abelian_Groups (but not yet imported by Algebra!)
paulson <lp15@cam.ac.uk>
parents: 70043
diff changeset
   291
  Note that despite the names \emph{mapping} and \emph{lookup} suggest something
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   292
  implementation-near, it is best to keep @{typ "'a \<Rightarrow>\<^sub>0 'b"} as an abstract
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   293
  \emph{algebraic} type
70045
7b6add80e3a5 fixed markup in Poly_Mapping; Free_Abelian_Groups (but not yet imported by Algebra!)
paulson <lp15@cam.ac.uk>
parents: 70043
diff changeset
   294
  providing operations like \emph{addition}, \emph{multiplication} without any notion
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   295
  of key-order, data structures etc.  This implementations-specific notions are
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   296
  easily introduced later for particular implementations but do not provide any
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   297
  gain for specifying logical properties of polynomials.
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   298
\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   299
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   300
subsection \<open>Additive structure\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   301
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   302
text \<open>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   303
  The additive structure covers the usual operations \<open>0\<close>, \<open>+\<close> and
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   304
  (unary and binary) \<open>-\<close>.  Recalling the ultimate interpretation, it
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   305
  is obvious that these have just lift the corresponding operations on values
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   306
  to mappings.
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   307
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   308
  Isabelle has a rich hierarchy of algebraic type classes, and in such situations
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   309
  of pointwise lifting a typical pattern is to have instantiations for a considerable
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   310
  number of type classes.
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   311
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   312
  The operations themselves are specified using \<open>lift_definition\<close>, where
70045
7b6add80e3a5 fixed markup in Poly_Mapping; Free_Abelian_Groups (but not yet imported by Algebra!)
paulson <lp15@cam.ac.uk>
parents: 70043
diff changeset
   313
  the proofs of the \emph{almost everywhere zero} property can be significantly involved.
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   314
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   315
  The @{const lookup} operation is supposed to be usable explicitly (unless in
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   316
  other situations where the morphisms between types are somehow internal
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   317
  to the \emph{lifting} package).  Hence it is good style to provide explicit rewrite
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   318
  rules how @{const lookup} acts on operations immediately.
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   319
\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   320
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   321
instantiation poly_mapping :: (type, zero) zero
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   322
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   323
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   324
lift_definition zero_poly_mapping :: "'a \<Rightarrow>\<^sub>0 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   325
  is "\<lambda>k. 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   326
  by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   327
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   328
instance ..
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   329
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   330
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   331
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   332
lemma Abs_poly_mapping [simp]: "Abs_poly_mapping (\<lambda>k. 0) = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   333
  by (simp add: zero_poly_mapping.abs_eq)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   334
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   335
lemma lookup_zero [simp]: "lookup 0 k = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   336
  by transfer rule
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   337
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   338
instantiation poly_mapping :: (type, monoid_add) monoid_add
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   339
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   340
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   341
lift_definition plus_poly_mapping ::
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   342
    "('a \<Rightarrow>\<^sub>0 'b) \<Rightarrow> ('a \<Rightarrow>\<^sub>0 'b) \<Rightarrow> 'a \<Rightarrow>\<^sub>0 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   343
  is "\<lambda>f1 f2 k. f1 k + f2 k"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   344
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   345
  fix f1 f2 :: "'a \<Rightarrow> 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   346
  assume "finite {k. f1 k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   347
    and "finite {k. f2 k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   348
  then have "finite ({k. f1 k \<noteq> 0} \<union> {k. f2 k \<noteq> 0})" by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   349
  moreover have "{x. f1 x + f2 x \<noteq> 0} \<subseteq> {k. f1 k \<noteq> 0} \<union> {k. f2 k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   350
    by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   351
  ultimately show "finite {x. f1 x + f2 x \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   352
    by (blast intro: finite_subset)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   353
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   354
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   355
instance
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   356
  by intro_classes (transfer, simp add: fun_eq_iff ac_simps)+
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   357
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   358
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   359
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
   360
lemma lookup_add: "lookup (f + g) k = lookup f k + lookup g k"
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
   361
  by (simp add: plus_poly_mapping.rep_eq)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   362
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   363
instance poly_mapping :: (type, comm_monoid_add) comm_monoid_add
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   364
  by intro_classes (transfer, simp add: fun_eq_iff ac_simps)+
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   365
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   366
lemma lookup_sum: "lookup (sum pp X) i = sum (\<lambda>x. lookup (pp x) i) X"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   367
  by (induction rule: infinite_finite_induct) (auto simp: lookup_add)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   368
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   369
(*instance poly_mapping :: (type, "{monoid_add, cancel_semigroup_add}") cancel_semigroup_add
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   370
  by intro_classes (transfer, simp add: fun_eq_iff)+*)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   371
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   372
instantiation poly_mapping :: (type, cancel_comm_monoid_add) cancel_comm_monoid_add
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   373
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   374
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   375
lift_definition minus_poly_mapping :: "('a \<Rightarrow>\<^sub>0 'b) \<Rightarrow> ('a \<Rightarrow>\<^sub>0 'b) \<Rightarrow> 'a \<Rightarrow>\<^sub>0 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   376
  is "\<lambda>f1 f2 k. f1 k - f2 k"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   377
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   378
  fix f1 f2 :: "'a \<Rightarrow> 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   379
  assume "finite {k. f1 k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   380
    and "finite {k. f2 k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   381
  then have "finite ({k. f1 k \<noteq> 0} \<union> {k. f2 k \<noteq> 0})" by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   382
  moreover have "{x. f1 x - f2 x \<noteq> 0} \<subseteq> {k. f1 k \<noteq> 0} \<union> {k. f2 k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   383
    by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   384
  ultimately show "finite {x. f1 x - f2 x \<noteq> 0}" by (blast intro: finite_subset)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   385
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   386
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   387
instance
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   388
  by intro_classes (transfer, simp add: fun_eq_iff diff_diff_add)+
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   389
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   390
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   391
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   392
instantiation poly_mapping :: (type, ab_group_add) ab_group_add
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   393
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   394
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   395
lift_definition uminus_poly_mapping :: "('a \<Rightarrow>\<^sub>0 'b) \<Rightarrow> 'a \<Rightarrow>\<^sub>0 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   396
  is uminus
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   397
  by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   398
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   399
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   400
instance
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   401
  by intro_classes (transfer, simp add: fun_eq_iff ac_simps)+
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   402
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   403
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   404
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   405
lemma lookup_uminus [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   406
  "lookup (- f) k = - lookup f k"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   407
  by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   408
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   409
lemma lookup_minus:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   410
  "lookup (f - g) k = lookup f k - lookup g k"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   411
  by transfer rule
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   412
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   413
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   414
subsection \<open>Multiplicative structure\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   415
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   416
instantiation poly_mapping :: (zero, zero_neq_one) zero_neq_one
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   417
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   418
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   419
lift_definition one_poly_mapping :: "'a \<Rightarrow>\<^sub>0 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   420
  is "\<lambda>k. 1 when k = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   421
  by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   422
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   423
instance
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   424
  by intro_classes (transfer, simp add: fun_eq_iff)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   425
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   426
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   427
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
   428
lemma lookup_one: "lookup 1 k = (1 when k = 0)"
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
   429
  by (meson one_poly_mapping.rep_eq)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   430
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   431
lemma lookup_one_zero [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   432
  "lookup 1 0 = 1"
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
   433
  by (simp add: one_poly_mapping.rep_eq)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   434
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   435
definition prod_fun :: "('a \<Rightarrow> 'b) \<Rightarrow> ('a \<Rightarrow> 'b) \<Rightarrow> 'a::monoid_add \<Rightarrow> 'b::semiring_0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   436
where
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   437
  "prod_fun f1 f2 k = (\<Sum>l. f1 l * (\<Sum>q. (f2 q when k = l + q)))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   438
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   439
lemma prod_fun_unfold_prod:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   440
  fixes f g :: "'a :: monoid_add \<Rightarrow> 'b::semiring_0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   441
  assumes fin_f: "finite {a. f a \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   442
  assumes fin_g: "finite {b. g b \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   443
  shows "prod_fun f g k = (\<Sum>(a, b). f a * g b when k = a + b)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   444
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   445
  let ?C = "{a. f a \<noteq> 0} \<times> {b. g b \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   446
  from fin_f fin_g have "finite ?C" by blast
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
   447
  moreover 
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
   448
  have "{a. \<exists>b. (f a * g b when k = a + b) \<noteq> 0} \<times>
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
   449
        {b. \<exists>a. (f a * g b when k = a + b) \<noteq> 0} \<subseteq> {a. f a \<noteq> 0} \<times> {b. g b \<noteq> 0}"
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   450
    by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   451
  ultimately show ?thesis using fin_g
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   452
    by (auto simp: prod_fun_def
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   453
      Sum_any.cartesian_product [of "{a. f a \<noteq> 0} \<times> {b. g b \<noteq> 0}"] Sum_any_right_distrib mult_when)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   454
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   455
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   456
lemma finite_prod_fun:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   457
  fixes f1 f2 :: "'a :: monoid_add \<Rightarrow> 'b :: semiring_0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   458
  assumes fin1: "finite {l. f1 l \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   459
  and fin2: "finite {q. f2 q \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   460
  shows "finite {k. prod_fun f1 f2 k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   461
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   462
  have *: "finite {k. (\<exists>l. f1 l \<noteq> 0 \<and> (\<exists>q. f2 q \<noteq> 0 \<and> k = l + q))}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   463
    using assms by simp
81332
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
   464
  have aux: "sum f2 {q. f2 q \<noteq> 0 \<and> k = l + q} = (\<Sum>q. (f2 q when k = l + q))" for k l
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
   465
  proof -
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   466
    have "{q. (f2 q when k = l + q) \<noteq> 0} \<subseteq> {q. f2 q \<noteq> 0 \<and> k = l + q}" by auto
81332
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
   467
    with fin2 show ?thesis
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
   468
      by (simp add: Sum_any.expand_superset [of "{q. f2 q \<noteq> 0 \<and> k = l + q}"])
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
   469
  qed
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   470
  have "{k. (\<Sum>l. f1 l * sum f2 {q. f2 q \<noteq> 0 \<and> k = l + q}) \<noteq> 0}
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   471
    \<subseteq> {k. (\<exists>l. f1 l * sum f2 {q. f2 q \<noteq> 0 \<and> k = l + q} \<noteq> 0)}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   472
    by (auto elim!: Sum_any.not_neutral_obtains_not_neutral)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   473
  also have "\<dots> \<subseteq> {k. (\<exists>l. f1 l \<noteq> 0 \<and> sum f2 {q. f2 q \<noteq> 0 \<and> k = l + q} \<noteq> 0)}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   474
    by (auto dest: mult_not_zero)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   475
  also have "\<dots> \<subseteq> {k. (\<exists>l. f1 l \<noteq> 0 \<and> (\<exists>q. f2 q \<noteq> 0 \<and> k = l + q))}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   476
    by (auto elim!: sum.not_neutral_contains_not_neutral)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   477
  finally have "finite {k. (\<Sum>l. f1 l * sum f2 {q. f2 q \<noteq> 0 \<and> k = l + q}) \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   478
    using * by (rule finite_subset)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   479
  with aux have "finite {k. (\<Sum>l. f1 l * (\<Sum>q. (f2 q when k = l + q))) \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   480
    by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   481
  with fin2 show ?thesis
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   482
   by (simp add: prod_fun_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   483
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   484
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   485
instantiation poly_mapping :: (monoid_add, semiring_0) semiring_0
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   486
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   487
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   488
lift_definition times_poly_mapping :: "('a \<Rightarrow>\<^sub>0 'b) \<Rightarrow> ('a \<Rightarrow>\<^sub>0 'b) \<Rightarrow> 'a \<Rightarrow>\<^sub>0 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   489
  is prod_fun
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   490
by(rule finite_prod_fun)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   491
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   492
instance
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   493
proof
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   494
  fix a b c :: "'a \<Rightarrow>\<^sub>0 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   495
  show "a * b * c = a * (b * c)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   496
  proof transfer
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   497
    fix f g h :: "'a \<Rightarrow> 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   498
    assume fin_f: "finite {a. f a \<noteq> 0}" (is "finite ?F")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   499
    assume fin_g: "finite {b. g b \<noteq> 0}" (is "finite ?G")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   500
    assume fin_h: "finite {c. h c \<noteq> 0}" (is "finite ?H")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   501
    from fin_f fin_g have fin_fg: "finite {(a, b). f a * g b \<noteq> 0}" (is "finite ?FG")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   502
      by (rule finite_mult_not_eq_zero_prodI)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   503
    from fin_g fin_h have fin_gh: "finite {(b, c). g b * h c \<noteq> 0}" (is "finite ?GH")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   504
      by (rule finite_mult_not_eq_zero_prodI)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   505
    from fin_f fin_g have fin_fg': "finite {a + b | a b. f a * g b \<noteq> 0}" (is "finite ?FG'")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   506
      by (rule finite_mult_not_eq_zero_sumI)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   507
    then have fin_fg'': "finite {d. (\<Sum>(a, b). f a * g b when d = a + b) \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   508
      by (auto intro: finite_Sum_any_not_eq_zero_weakenI)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   509
    from fin_g fin_h have fin_gh': "finite {b + c | b c. g b * h c \<noteq> 0}" (is "finite ?GH'")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   510
      by (rule finite_mult_not_eq_zero_sumI)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   511
    then have fin_gh'': "finite {d. (\<Sum>(b, c). g b * h c when d = b + c) \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   512
      by (auto intro: finite_Sum_any_not_eq_zero_weakenI)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   513
    show "prod_fun (prod_fun f g) h = prod_fun f (prod_fun g h)" (is "?lhs = ?rhs")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   514
    proof
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   515
      fix k
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   516
      from fin_f fin_g fin_h fin_fg''
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   517
      have "?lhs k = (\<Sum>(ab, c). (\<Sum>(a, b). f a * g b when ab = a + b) * h c when k = ab + c)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   518
        by (simp add: prod_fun_unfold_prod)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   519
      also have "\<dots> = (\<Sum>(ab, c). (\<Sum>(a, b). f a * g b * h c when k = ab + c when ab = a + b))"
81332
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
   520
        using fin_fg
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   521
        apply (simp add: Sum_any_left_distrib split_def flip: Sum_any_when_independent)
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   522
        apply (simp add: when_when when_mult mult_when conj_commute)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   523
        done
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   524
      also have "\<dots> = (\<Sum>(ab, c, a, b). f a * g b * h c when k = ab + c when ab = a + b)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   525
        apply (subst Sum_any.cartesian_product2 [of "(?FG' \<times> ?H) \<times> ?FG"])
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   526
        apply (auto simp: finite_cartesian_product_iff fin_fg fin_fg' fin_h dest: mult_not_zero)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   527
        done
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   528
      also have "\<dots> = (\<Sum>(ab, c, a, b). f a * g b * h c when k = a + b + c when ab = a + b)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   529
        by (rule Sum_any.cong) (simp add: split_def when_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   530
      also have "\<dots> = (\<Sum>(ab, cab). (case cab of (c, a, b) \<Rightarrow> f a * g b * h c when k = a + b + c)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   531
        when ab = (case cab of (c, a, b) \<Rightarrow> a + b))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   532
        by (simp add: split_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   533
      also have "\<dots> = (\<Sum>(c, a, b). f a * g b * h c when k = a + b + c)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   534
        by (simp add: Sum_any_when_dependent_prod_left)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   535
      also have "\<dots> = (\<Sum>(bc, cab). (case cab of (c, a, b) \<Rightarrow> f a * g b * h c when k = a + b + c)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   536
        when bc = (case cab of (c, a, b) \<Rightarrow> b + c))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   537
        by (simp add: Sum_any_when_dependent_prod_left)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   538
      also have "\<dots> = (\<Sum>(bc, c, a, b). f a * g b * h c when k = a + b + c when bc = b + c)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   539
        by (simp add: split_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   540
      also have "\<dots> = (\<Sum>(bc, c, a, b). f a * g b * h c when bc = b + c when k = a + bc)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   541
        by (rule Sum_any.cong) (simp add: split_def when_def ac_simps)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   542
      also have "\<dots> = (\<Sum>(a, bc, b, c). f a * g b * h c when bc = b + c when k = a + bc)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   543
      proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   544
        have "bij (\<lambda>(a, d, b, c). (d, c, a, b))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   545
          by (auto intro!: bijI injI surjI [of _ "\<lambda>(d, c, a, b). (a, d, b, c)"] simp add: split_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   546
        then show ?thesis
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   547
          by (rule Sum_any.reindex_cong) auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   548
      qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   549
      also have "\<dots> = (\<Sum>(a, bc). (\<Sum>(b, c). f a * g b * h c when bc = b + c when k = a + bc))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   550
        apply (subst Sum_any.cartesian_product2 [of "(?F \<times> ?GH') \<times> ?GH"])
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   551
        apply (auto simp: finite_cartesian_product_iff fin_f fin_gh fin_gh' ac_simps dest: mult_not_zero)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   552
        done
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   553
      also have "\<dots> = (\<Sum>(a, bc). f a * (\<Sum>(b, c). g b * h c when bc = b + c) when k = a + bc)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   554
        apply (subst Sum_any_right_distrib)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   555
        using fin_gh apply (simp add: split_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   556
        apply (subst Sum_any_when_independent [symmetric])
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   557
        apply (simp add: when_when when_mult mult_when split_def ac_simps)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   558
        done
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   559
      also from fin_f fin_g fin_h fin_gh''
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   560
      have "\<dots> = ?rhs k"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   561
        by (simp add: prod_fun_unfold_prod)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   562
      finally show "?lhs k = ?rhs k" .
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   563
    qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   564
  qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   565
  show "(a + b) * c = a * c + b * c"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   566
  proof transfer
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   567
    fix f g h :: "'a \<Rightarrow> 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   568
    assume fin_f: "finite {k. f k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   569
    assume fin_g: "finite {k. g k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   570
    assume fin_h: "finite {k. h k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   571
    show "prod_fun (\<lambda>k. f k + g k) h = (\<lambda>k. prod_fun f h k + prod_fun g h k)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   572
      apply (rule ext)
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   573
      apply (simp add: prod_fun_def algebra_simps)
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   574
      by (simp add: Sum_any.distrib fin_f fin_g finite_mult_not_eq_zero_rightI)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   575
  qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   576
  show "a * (b + c) = a * b + a * c"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   577
  proof transfer
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   578
    fix f g h :: "'a \<Rightarrow> 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   579
    assume fin_f: "finite {k. f k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   580
    assume fin_g: "finite {k. g k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   581
    assume fin_h: "finite {k. h k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   582
    show "prod_fun f (\<lambda>k. g k + h k) = (\<lambda>k. prod_fun f g k + prod_fun f h k)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   583
      apply (rule ext)
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   584
      apply (auto simp: prod_fun_def Sum_any.distrib algebra_simps when_add_distrib fin_g fin_h)
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   585
      by (simp add: Sum_any.distrib fin_f finite_mult_not_eq_zero_rightI)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   586
  qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   587
  show "0 * a = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   588
    by transfer (simp add: prod_fun_def [abs_def])
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   589
  show "a * 0 = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   590
    by transfer (simp add: prod_fun_def [abs_def])
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   591
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   592
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   593
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   594
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   595
lemma lookup_mult:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   596
  "lookup (f * g) k = (\<Sum>l. lookup f l * (\<Sum>q. lookup g q when k = l + q))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   597
  by transfer (simp add: prod_fun_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   598
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   599
instance poly_mapping :: (comm_monoid_add, comm_semiring_0) comm_semiring_0
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   600
proof
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   601
  fix a b c :: "'a \<Rightarrow>\<^sub>0 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   602
  show "a * b = b * a"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   603
  proof transfer
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   604
    fix f g :: "'a \<Rightarrow> 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   605
    assume fin_f: "finite {a. f a \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   606
    assume fin_g: "finite {b. g b \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   607
    show "prod_fun f g = prod_fun g f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   608
    proof
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   609
      fix k
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   610
      have fin1: "\<And>l. finite {a. (f a when k = l + a) \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   611
        using fin_f by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   612
      have fin2: "\<And>l. finite {b. (g b when k = l + b) \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   613
        using fin_g by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   614
      from fin_f fin_g have "finite {(a, b). f a \<noteq> 0 \<and> g b \<noteq> 0}" (is "finite ?AB")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   615
        by simp
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   616
      have "(\<Sum>a. \<Sum>n. f a * (g n when k = a + n)) = (\<Sum>a. \<Sum>n. g a * (f n when k = a + n))"
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   617
        by (subst Sum_any.swap [OF \<open>finite ?AB\<close>]) (auto simp: mult_when ac_simps)
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   618
      then show "prod_fun f g k = prod_fun g f k"
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   619
        by (simp add: prod_fun_def Sum_any_right_distrib [OF fin2] Sum_any_right_distrib [OF fin1])
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   620
    qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   621
  qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   622
  show "(a + b) * c = a * c + b * c"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   623
  proof transfer
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   624
    fix f g h :: "'a \<Rightarrow> 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   625
    assume fin_f: "finite {k. f k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   626
    assume fin_g: "finite {k. g k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   627
    assume fin_h: "finite {k. h k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   628
    show "prod_fun (\<lambda>k. f k + g k) h = (\<lambda>k. prod_fun f h k + prod_fun g h k)"
81332
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
   629
      by (auto simp: prod_fun_def fun_eq_iff algebra_simps
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   630
            Sum_any.distrib fin_f fin_g finite_mult_not_eq_zero_rightI)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   631
  qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   632
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   633
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   634
instance poly_mapping :: (monoid_add, semiring_0_cancel) semiring_0_cancel
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   635
  ..
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   636
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   637
instance poly_mapping :: (comm_monoid_add, comm_semiring_0_cancel) comm_semiring_0_cancel
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   638
  ..
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   639
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   640
instance poly_mapping :: (monoid_add, semiring_1) semiring_1
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   641
proof
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   642
  fix a :: "'a \<Rightarrow>\<^sub>0 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   643
  show "1 * a = a"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   644
    by transfer (simp add: prod_fun_def [abs_def] when_mult)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   645
  show "a * 1 = a"
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
   646
    apply transfer 
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   647
    apply (simp add: prod_fun_def [abs_def] Sum_any_right_distrib Sum_any_left_distrib mult_when)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   648
    apply (subst when_commute)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   649
    apply simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   650
    done
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   651
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   652
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   653
instance poly_mapping :: (comm_monoid_add, comm_semiring_1) comm_semiring_1
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   654
proof
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   655
  fix a :: "'a \<Rightarrow>\<^sub>0 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   656
  show "1 * a = a"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   657
    by transfer (simp add: prod_fun_def [abs_def])
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   658
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   659
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   660
instance poly_mapping :: (monoid_add, semiring_1_cancel) semiring_1_cancel
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   661
  ..
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   662
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   663
instance poly_mapping :: (monoid_add, ring) ring
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   664
  ..
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   665
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   666
instance poly_mapping :: (comm_monoid_add, comm_ring) comm_ring
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   667
  ..
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   668
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   669
instance poly_mapping :: (monoid_add, ring_1) ring_1
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   670
  ..
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   671
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   672
instance poly_mapping :: (comm_monoid_add, comm_ring_1) comm_ring_1
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   673
  ..
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   674
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   675
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   676
subsection \<open>Single-point mappings\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   677
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   678
lift_definition single :: "'a \<Rightarrow> 'b \<Rightarrow> 'a \<Rightarrow>\<^sub>0 'b::zero"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   679
  is "\<lambda>k v k'. (v when k = k')"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   680
  by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   681
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   682
lemma inj_single [iff]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   683
  "inj (single k)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   684
proof (rule injI, transfer)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   685
  fix k :: 'b and a b :: "'a::zero"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   686
  assume "(\<lambda>k'. a when k = k') = (\<lambda>k'. b when k = k')"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   687
  then have "(\<lambda>k'. a when k = k') k = (\<lambda>k'. b when k = k') k"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   688
    by (rule arg_cong)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   689
  then show "a = b" by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   690
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   691
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   692
lemma lookup_single:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   693
  "lookup (single k v) k' = (v when k = k')"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   694
  by (simp add: single.rep_eq)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   695
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   696
lemma lookup_single_eq [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   697
  "lookup (single k v) k = v"
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
   698
  by (simp add: single.rep_eq)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   699
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   700
lemma lookup_single_not_eq:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   701
  "k \<noteq> k' \<Longrightarrow> lookup (single k v) k' = 0"
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
   702
  by (simp add: single.rep_eq)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   703
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   704
lemma single_zero [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   705
  "single k 0 = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   706
  by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   707
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   708
lemma single_one [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   709
  "single 0 1 = 1"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   710
  by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   711
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   712
lemma single_add:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   713
  "single k (a + b) = single k a + single k b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   714
  by transfer (simp add: fun_eq_iff when_add_distrib)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   715
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   716
lemma single_uminus:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   717
  "single k (- a) = - single k a"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   718
  by transfer (simp add: fun_eq_iff when_uminus_distrib)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   719
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   720
lemma single_diff:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   721
  "single k (a - b) = single k a - single k b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   722
  by transfer (simp add: fun_eq_iff when_diff_distrib)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   723
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   724
lemma single_numeral [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   725
  "single 0 (numeral n) = numeral n"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   726
  by (induct n) (simp_all only: numeral.simps numeral_add single_zero single_one single_add)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   727
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   728
lemma lookup_numeral:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   729
  "lookup (numeral n) k = (numeral n when k = 0)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   730
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   731
  have "lookup (numeral n) k = lookup (single 0 (numeral n)) k"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   732
    by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   733
  then show ?thesis unfolding lookup_single by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   734
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   735
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   736
lemma single_of_nat [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   737
  "single 0 (of_nat n) = of_nat n"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   738
  by (induct n) (simp_all add: single_add)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   739
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   740
lemma lookup_of_nat:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   741
  "lookup (of_nat n) k = (of_nat n when k = 0)"
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
   742
  by (metis lookup_single lookup_single_not_eq single_of_nat)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   743
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   744
lemma of_nat_single:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   745
  "of_nat = single 0 \<circ> of_nat"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   746
  by (simp add: fun_eq_iff)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   747
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   748
lemma mult_single:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   749
  "single k a * single l b = single (k + l) (a * b)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   750
proof transfer
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   751
  fix k l :: 'a and a b :: 'b
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   752
  show "prod_fun (\<lambda>k'. a when k = k') (\<lambda>k'. b when l = k') = (\<lambda>k'. a * b when k + l = k')"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   753
  proof
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   754
    fix k'
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   755
    have "prod_fun (\<lambda>k'. a when k = k') (\<lambda>k'. b when l = k') k' = (\<Sum>n. a * b when l = n when k' = k + n)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   756
      by (simp add: prod_fun_def Sum_any_right_distrib mult_when when_mult)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   757
    also have "\<dots> = (\<Sum>n. a * b when k' = k + n when l = n)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   758
      by (simp add: when_when conj_commute)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   759
    also have "\<dots> = (a * b when k' = k + l)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   760
      by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   761
    also have "\<dots> = (a * b when k + l = k')"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   762
      by (simp add: when_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   763
    finally show "prod_fun (\<lambda>k'. a when k = k') (\<lambda>k'. b when l = k') k' =
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   764
      (\<lambda>k'. a * b when k + l = k') k'" .
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   765
  qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   766
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   767
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   768
instance poly_mapping :: (monoid_add, semiring_char_0) semiring_char_0
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   769
  by intro_classes (auto intro: inj_compose inj_of_nat simp add: of_nat_single)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   770
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   771
instance poly_mapping :: (monoid_add, ring_char_0) ring_char_0
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   772
  ..
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   773
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   774
lemma single_of_int [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   775
  "single 0 (of_int k) = of_int k"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   776
  by (cases k) (simp_all add: single_diff single_uminus)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   777
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   778
lemma lookup_of_int:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   779
  "lookup (of_int l) k = (of_int l when k = 0)"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   780
  by (metis lookup_single_not_eq single.rep_eq single_of_int)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   781
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   782
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   783
subsection \<open>Integral domains\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   784
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   785
instance poly_mapping :: ("{ordered_cancel_comm_monoid_add, linorder}", semiring_no_zero_divisors) semiring_no_zero_divisors
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   786
  text \<open>The @{class "linorder"} constraint is a pragmatic device for the proof --- maybe it can be dropped\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   787
proof
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   788
  fix f g :: "'a \<Rightarrow>\<^sub>0 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   789
  assume "f \<noteq> 0" and "g \<noteq> 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   790
  then show "f * g \<noteq> 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   791
  proof transfer
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   792
    fix f g :: "'a \<Rightarrow> 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   793
    define F where "F = {a. f a \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   794
    moreover define G where "G = {a. g a \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   795
    ultimately have [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   796
      "\<And>a. f a \<noteq> 0 \<longleftrightarrow> a \<in> F"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   797
      "\<And>b. g b \<noteq> 0 \<longleftrightarrow> b \<in> G"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   798
      by simp_all
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   799
    assume "finite {a. f a \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   800
    then have [simp]: "finite F"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   801
      by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   802
    assume "finite {a. g a \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   803
    then have [simp]: "finite G"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   804
      by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   805
    assume "f \<noteq> (\<lambda>a. 0)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   806
    then obtain a where "f a \<noteq> 0"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   807
      by (auto simp: fun_eq_iff)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   808
    assume "g \<noteq> (\<lambda>b. 0)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   809
    then obtain b where "g b \<noteq> 0"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   810
      by (auto simp: fun_eq_iff)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   811
    from \<open>f a \<noteq> 0\<close> and \<open>g b \<noteq> 0\<close> have "F \<noteq> {}" and "G \<noteq> {}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   812
      by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   813
    note Max_F = \<open>finite F\<close> \<open>F \<noteq> {}\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   814
    note Max_G = \<open>finite G\<close> \<open>G \<noteq> {}\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   815
    from Max_F and Max_G have [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   816
      "Max F \<in> F"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   817
      "Max G \<in> G"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   818
      by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   819
    from Max_F Max_G have [dest!]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   820
      "\<And>a. a \<in> F \<Longrightarrow> a \<le> Max F"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   821
      "\<And>b. b \<in> G \<Longrightarrow> b \<le> Max G"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   822
      by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   823
    define q where "q = Max F + Max G"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   824
    have "(\<Sum>(a, b). f a * g b when q = a + b) =
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   825
      (\<Sum>(a, b). f a * g b when q = a + b when a \<in> F \<and> b \<in> G)"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   826
      by (rule Sum_any.cong) (auto simp: split_def when_def q_def intro: ccontr)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   827
    also have "\<dots> =
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   828
      (\<Sum>(a, b). f a * g b when (Max F, Max G) = (a, b))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   829
    proof (rule Sum_any.cong)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   830
      fix ab :: "'a \<times> 'a"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   831
      obtain a b where [simp]: "ab = (a, b)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   832
        by (cases ab) simp_all
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   833
      have [dest!]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   834
        "a \<le> Max F \<Longrightarrow> Max F \<noteq> a \<Longrightarrow> a < Max F"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   835
        "b \<le> Max G \<Longrightarrow> Max G \<noteq> b \<Longrightarrow> b < Max G"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   836
        by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   837
      show "(case ab of (a, b) \<Rightarrow> f a * g b when q = a + b when a \<in> F \<and> b \<in> G) =
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   838
         (case ab of (a, b) \<Rightarrow> f a * g b when (Max F, Max G) = (a, b))"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   839
        by (auto simp: split_def when_def q_def dest: add_strict_mono [of a "Max F" b "Max G"])
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   840
    qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   841
    also have "\<dots> = (\<Sum>ab. (case ab of (a, b) \<Rightarrow> f a * g b) when
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   842
      (Max F, Max G) = ab)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   843
      unfolding split_def when_def by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   844
    also have "\<dots> \<noteq> 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   845
      by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   846
    finally have "prod_fun f g q \<noteq> 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   847
      by (simp add: prod_fun_unfold_prod)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   848
    then show "prod_fun f g \<noteq> (\<lambda>k. 0)"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   849
      by (auto simp: fun_eq_iff)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   850
  qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   851
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   852
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   853
instance poly_mapping :: ("{ordered_cancel_comm_monoid_add, linorder}", ring_no_zero_divisors) ring_no_zero_divisors
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   854
  ..
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   855
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   856
instance poly_mapping :: ("{ordered_cancel_comm_monoid_add, linorder}", ring_1_no_zero_divisors) ring_1_no_zero_divisors
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   857
  ..
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   858
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   859
instance poly_mapping :: ("{ordered_cancel_comm_monoid_add, linorder}", idom) idom
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   860
  ..
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   861
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   862
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   863
subsection \<open>Mapping order\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   864
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   865
instantiation poly_mapping :: (linorder, "{zero, linorder}") linorder
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   866
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   867
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   868
lift_definition less_poly_mapping :: "('a \<Rightarrow>\<^sub>0 'b) \<Rightarrow> ('a \<Rightarrow>\<^sub>0 'b) \<Rightarrow> bool"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   869
  is less_fun
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   870
  .
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   871
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   872
lift_definition less_eq_poly_mapping :: "('a \<Rightarrow>\<^sub>0 'b) \<Rightarrow> ('a \<Rightarrow>\<^sub>0 'b) \<Rightarrow> bool"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   873
  is "\<lambda>f g. less_fun f g \<or> f = g"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   874
  .
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   875
77955
c4677a6aae2c more standard name bindings (amending 5bf71b4da706): avoid odd full_name like "Orderings.class.Orderings.preorder.of_class.intro" with many redundant name space accesses;
wenzelm
parents: 76987
diff changeset
   876
instance proof (rule linorder.intro_of_class)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   877
  show "class.linorder (less_eq :: (_ \<Rightarrow>\<^sub>0 _) \<Rightarrow> _) less"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   878
  proof (rule linorder_strictI, rule order_strictI)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   879
    fix f g h :: "'a \<Rightarrow>\<^sub>0 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   880
    show "f \<le> g \<longleftrightarrow> f < g \<or> f = g"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   881
      by transfer (rule refl)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   882
    show "\<not> f < f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   883
      by transfer (rule less_fun_irrefl)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   884
    show "f < g \<or> f = g \<or> g < f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   885
    proof transfer
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   886
      fix f g :: "'a \<Rightarrow> 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   887
      assume "finite {k. f k \<noteq> 0}" and "finite {k. g k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   888
      then have "finite ({k. f k \<noteq> 0} \<union> {k. g k \<noteq> 0})"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   889
        by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   890
      moreover have "{k. f k \<noteq> g k} \<subseteq> {k. f k \<noteq> 0} \<union> {k. g k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   891
        by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   892
      ultimately have "finite {k. f k \<noteq> g k}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   893
        by (rule rev_finite_subset)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   894
      then show "less_fun f g \<or> f = g \<or> less_fun g f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   895
        by (rule less_fun_trichotomy)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   896
    qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   897
    assume "f < g" then show "\<not> g < f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   898
      by transfer (rule less_fun_asym)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   899
    note \<open>f < g\<close> moreover assume "g < h"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   900
      ultimately show "f < h"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   901
      by transfer (rule less_fun_trans)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   902
  qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   903
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   904
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   905
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   906
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   907
instance poly_mapping :: (linorder, "{ordered_comm_monoid_add, ordered_ab_semigroup_add_imp_le, linorder}") ordered_ab_semigroup_add
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   908
proof (intro_classes, transfer)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   909
  fix f g h :: "'a \<Rightarrow> 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   910
  assume *: "less_fun f g \<or> f = g"
81332
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
   911
  have "less_fun (\<lambda>k. h k + f k) (\<lambda>k. h k + g k)" if "less_fun f g"
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
   912
    by (metis (no_types, lifting) less_fun_def add_strict_left_mono that)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   913
  with * show "less_fun (\<lambda>k. h k + f k) (\<lambda>k. h k + g k) \<or> (\<lambda>k. h k + f k) = (\<lambda>k. h k + g k)"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   914
    by (auto simp: fun_eq_iff)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   915
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   916
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   917
instance poly_mapping :: (linorder, "{ordered_comm_monoid_add, ordered_ab_semigroup_add_imp_le, cancel_comm_monoid_add, linorder}") linordered_cancel_ab_semigroup_add
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   918
  ..
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   919
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   920
instance poly_mapping :: (linorder, "{ordered_comm_monoid_add, ordered_ab_semigroup_add_imp_le, cancel_comm_monoid_add, linorder}") ordered_comm_monoid_add
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   921
  ..
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   922
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   923
instance poly_mapping :: (linorder, "{ordered_comm_monoid_add, ordered_ab_semigroup_add_imp_le, cancel_comm_monoid_add, linorder}") ordered_cancel_comm_monoid_add
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   924
  ..
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   925
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   926
instance poly_mapping :: (linorder, linordered_ab_group_add) linordered_ab_group_add
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   927
  ..
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   928
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   929
text \<open>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   930
  For pragmatism we leave out the final elements in the hierarchy:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   931
  @{class linordered_ring}, @{class linordered_ring_strict}, @{class linordered_idom};
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   932
  remember that the order instance is a mere technical device, not a deeper algebraic
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   933
  property.
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   934
\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   935
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   936
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   937
subsection \<open>Fundamental mapping notions\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   938
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   939
lift_definition keys :: "('a \<Rightarrow>\<^sub>0 'b::zero) \<Rightarrow> 'a set"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   940
  is "\<lambda>f. {k. f k \<noteq> 0}" .
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   941
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   942
lift_definition range :: "('a \<Rightarrow>\<^sub>0 'b::zero) \<Rightarrow> 'b set"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   943
  is "\<lambda>f :: 'a \<Rightarrow> 'b. Set.range f - {0}" .
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   944
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   945
lemma finite_keys [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   946
  "finite (keys f)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   947
  by transfer
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   948
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   949
lemma not_in_keys_iff_lookup_eq_zero:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   950
  "k \<notin> keys f \<longleftrightarrow> lookup f k = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   951
  by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   952
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   953
lemma lookup_not_eq_zero_eq_in_keys:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   954
  "lookup f k \<noteq> 0 \<longleftrightarrow> k \<in> keys f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   955
  by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   956
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   957
lemma lookup_eq_zero_in_keys_contradict [dest]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   958
  "lookup f k = 0 \<Longrightarrow> \<not> k \<in> keys f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   959
  by (simp add: not_in_keys_iff_lookup_eq_zero)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   960
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   961
lemma finite_range [simp]: "finite (Poly_Mapping.range p)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   962
proof transfer
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   963
  fix f :: "'b \<Rightarrow> 'a"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   964
  assume *: "finite {x. f x \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   965
  have "Set.range f - {0} \<subseteq> f ` {x. f x \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   966
    by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   967
  thus "finite (Set.range f - {0})"
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
   968
    using "*" finite_surj by blast
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   969
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   970
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   971
lemma in_keys_lookup_in_range [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   972
  "k \<in> keys f \<Longrightarrow> lookup f k \<in> range f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   973
  by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   974
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   975
lemma in_keys_iff: "x \<in> (keys s) = (lookup s x \<noteq> 0)"
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
   976
  by (simp add: lookup_not_eq_zero_eq_in_keys)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   977
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   978
lemma keys_zero [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   979
  "keys 0 = {}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   980
  by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   981
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   982
lemma range_zero [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   983
  "range 0 = {}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   984
  by transfer auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   985
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   986
lemma keys_add:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   987
  "keys (f + g) \<subseteq> keys f \<union> keys g"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   988
  by transfer auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   989
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   990
lemma keys_one [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   991
  "keys 1 = {0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   992
  by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   993
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   994
lemma range_one [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   995
  "range 1 = {1}"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
   996
  by transfer (auto simp: when_def)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   997
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   998
lemma keys_single [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
   999
  "keys (single k v) = (if v = 0 then {} else {k})"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1000
  by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1001
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1002
lemma range_single [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1003
  "range (single k v) = (if v = 0 then {} else {v})"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1004
  by transfer (auto simp: when_def)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1005
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1006
lemma keys_mult:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1007
  "keys (f * g) \<subseteq> {a + b | a b. a \<in> keys f \<and> b \<in> keys g}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1008
  apply transfer
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1009
  apply (force simp: prod_fun_def dest!: mult_not_zero elim!: Sum_any.not_neutral_obtains_not_neutral)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1010
  done
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1011
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1012
lemma setsum_keys_plus_distrib:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1013
  assumes hom_0: "\<And>k. f k 0 = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1014
  and hom_plus: "\<And>k. k \<in> Poly_Mapping.keys p \<union> Poly_Mapping.keys q \<Longrightarrow> f k (Poly_Mapping.lookup p k + Poly_Mapping.lookup q k) = f k (Poly_Mapping.lookup p k) + f k (Poly_Mapping.lookup q k)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1015
  shows
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1016
  "(\<Sum>k\<in>Poly_Mapping.keys (p + q). f k (Poly_Mapping.lookup (p + q) k)) =
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1017
   (\<Sum>k\<in>Poly_Mapping.keys p. f k (Poly_Mapping.lookup p k)) +
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1018
   (\<Sum>k\<in>Poly_Mapping.keys q. f k (Poly_Mapping.lookup q k))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1019
  (is "?lhs = ?p + ?q")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1020
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1021
  let ?A = "Poly_Mapping.keys p \<union> Poly_Mapping.keys q"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1022
  have "?lhs = (\<Sum>k\<in>?A. f k (Poly_Mapping.lookup p k + Poly_Mapping.lookup q k))"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1023
    by(intro sum.mono_neutral_cong_left) (auto simp: sum.mono_neutral_cong_left hom_0 in_keys_iff lookup_add)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1024
  also have "\<dots> = (\<Sum>k\<in>?A. f k (Poly_Mapping.lookup p k) + f k (Poly_Mapping.lookup q k))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1025
    by(rule sum.cong)(simp_all add: hom_plus)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1026
  also have "\<dots> = (\<Sum>k\<in>?A. f k (Poly_Mapping.lookup p k)) + (\<Sum>k\<in>?A. f k (Poly_Mapping.lookup q k))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1027
    (is "_ = ?p' + ?q'")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1028
    by(simp add: sum.distrib)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1029
  also have "?p' = ?p"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1030
    by (simp add: hom_0 in_keys_iff sum.mono_neutral_cong_right)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1031
  also have "?q' = ?q"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1032
    by (simp add: hom_0 in_keys_iff sum.mono_neutral_cong_right)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1033
  finally show ?thesis .
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1034
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1035
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1036
subsection \<open>Degree\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1037
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1038
definition degree :: "(nat \<Rightarrow>\<^sub>0 'a::zero) \<Rightarrow> nat"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1039
where
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1040
  "degree f = Max (insert 0 (Suc ` keys f))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1041
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1042
lemma degree_zero [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1043
  "degree 0 = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1044
  unfolding degree_def by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1045
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1046
lemma degree_one [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1047
  "degree 1 = 1"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1048
  unfolding degree_def by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1049
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1050
lemma degree_single_zero [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1051
  "degree (single k 0) = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1052
  unfolding degree_def by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1053
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1054
lemma degree_single_not_zero [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1055
  "v \<noteq> 0 \<Longrightarrow> degree (single k v) = Suc k"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1056
  unfolding degree_def by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1057
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1058
lemma degree_zero_iff [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1059
  "degree f = 0 \<longleftrightarrow> f = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1060
unfolding degree_def proof transfer
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1061
  fix f :: "nat \<Rightarrow> 'a"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1062
  assume "finite {n. f n \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1063
  then have fin: "finite (insert 0 (Suc ` {n. f n \<noteq> 0}))" by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1064
  show "Max (insert 0 (Suc ` {n. f n \<noteq> 0})) = 0 \<longleftrightarrow> f = (\<lambda>n. 0)" (is "?P \<longleftrightarrow> ?Q")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1065
  proof
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1066
    assume ?P
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1067
    have "{n. f n \<noteq> 0} = {}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1068
    proof (rule ccontr)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1069
      assume "{n. f n \<noteq> 0} \<noteq> {}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1070
      then obtain n where "n \<in> {n. f n \<noteq> 0}" by blast
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1071
      then have "{n. f n \<noteq> 0} = insert n {n. f n \<noteq> 0}" by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1072
      then have "Suc ` {n. f n \<noteq> 0} = insert (Suc n) (Suc ` {n. f n \<noteq> 0})" by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1073
      with \<open>?P\<close> have "Max (insert 0 (insert (Suc n) (Suc ` {n. f n \<noteq> 0}))) = 0" by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1074
      then have "Max (insert (Suc n) (insert 0 (Suc ` {n. f n \<noteq> 0}))) = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1075
        by (simp add: insert_commute)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1076
      with fin have "max (Suc n) (Max (insert 0 (Suc ` {n. f n \<noteq> 0}))) = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1077
        by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1078
      then show False by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1079
    qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1080
    then show ?Q by (simp add: fun_eq_iff)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1081
  next
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1082
    assume ?Q then show ?P by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1083
  qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1084
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1085
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1086
lemma degree_greater_zero_in_keys:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1087
  assumes "0 < degree f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1088
  shows "degree f - 1 \<in> keys f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1089
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1090
  from assms have "keys f \<noteq> {}"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1091
    by (auto simp: degree_def)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1092
  then show ?thesis unfolding degree_def
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1093
    by (simp add: mono_Max_commute [symmetric] mono_Suc)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1094
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1095
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1096
lemma in_keys_less_degree:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1097
  "n \<in> keys f \<Longrightarrow> n < degree f"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1098
unfolding degree_def by transfer (auto simp: Max_gr_iff)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1099
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1100
lemma beyond_degree_lookup_zero:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1101
  "degree f \<le> n \<Longrightarrow> lookup f n = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1102
  unfolding degree_def by transfer auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1103
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1104
lemma degree_add:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1105
  "degree (f + g) \<le> max (degree f) (Poly_Mapping.degree g)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1106
unfolding degree_def proof transfer
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1107
  fix f g :: "nat \<Rightarrow> 'a"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1108
  assume f: "finite {x. f x \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1109
  assume g: "finite {x. g x \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1110
  let ?f = "Max (insert 0 (Suc ` {k. f k \<noteq> 0}))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1111
  let ?g = "Max (insert 0 (Suc ` {k. g k \<noteq> 0}))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1112
  have "Max (insert 0 (Suc ` {k. f k + g k \<noteq> 0})) \<le> Max (insert 0 (Suc ` ({k. f k \<noteq> 0} \<union> {k. g k \<noteq> 0})))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1113
    by (rule Max.subset_imp) (insert f g, auto)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1114
  also have "\<dots> = max ?f ?g"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1115
    using f g by (simp_all add: image_Un Max_Un [symmetric])
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1116
  finally show "Max (insert 0 (Suc ` {k. f k + g k \<noteq> 0}))
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1117
    \<le> max (Max (insert 0 (Suc ` {k. f k \<noteq> 0}))) (Max (insert 0 (Suc ` {k. g k \<noteq> 0})))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1118
    .
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1119
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1120
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1121
lemma sorted_list_of_set_keys:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1122
  "sorted_list_of_set (keys f) = filter (\<lambda>k. k \<in> keys f) [0..<degree f]" (is "_ = ?r")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1123
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1124
  have "keys f = set ?r"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1125
    by (auto dest: in_keys_less_degree)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1126
  moreover have "sorted_list_of_set (set ?r) = ?r"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1127
    unfolding sorted_list_of_set_sort_remdups
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1128
    by (simp add: remdups_filter filter_sort [symmetric])
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1129
  ultimately show ?thesis by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1130
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1131
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1132
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1133
subsection \<open>Inductive structure\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1134
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1135
lift_definition update :: "'a \<Rightarrow> 'b \<Rightarrow> ('a \<Rightarrow>\<^sub>0 'b::zero) \<Rightarrow> 'a \<Rightarrow>\<^sub>0 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1136
  is "\<lambda>k v f. f(k := v)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1137
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1138
  fix f :: "'a \<Rightarrow> 'b" and k' v
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1139
  assume "finite {k. f k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1140
  then have "finite (insert k' {k. f k \<noteq> 0})"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1141
    by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1142
  then show "finite {k. (f(k' := v)) k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1143
    by (rule rev_finite_subset) auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1144
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1145
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1146
lemma update_induct [case_names const update]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1147
  assumes const': "P 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1148
  assumes update': "\<And>f a b. a \<notin> keys f \<Longrightarrow> b \<noteq> 0 \<Longrightarrow> P f \<Longrightarrow> P (update a b f)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1149
  shows "P f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1150
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1151
  obtain g where "f = Abs_poly_mapping g" and "finite {a. g a \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1152
    by (cases f) simp_all
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1153
  define Q where "Q g = P (Abs_poly_mapping g)" for g
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1154
  from \<open>finite {a. g a \<noteq> 0}\<close> have "Q g"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1155
  proof (induct g rule: finite_update_induct)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1156
    case const with const' Q_def show ?case
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1157
      by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1158
  next
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1159
    case (update a b g)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1160
    from \<open>finite {a. g a \<noteq> 0}\<close> \<open>g a = 0\<close> have "a \<notin> keys (Abs_poly_mapping g)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1161
      by (simp add: Abs_poly_mapping_inverse keys.rep_eq)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1162
    moreover note \<open>b \<noteq> 0\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1163
    moreover from \<open>Q g\<close> have "P (Abs_poly_mapping g)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1164
      by (simp add: Q_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1165
    ultimately have "P (update a b (Abs_poly_mapping g))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1166
      by (rule update')
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1167
    also from \<open>finite {a. g a \<noteq> 0}\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1168
    have "update a b (Abs_poly_mapping g) = Abs_poly_mapping (g(a := b))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1169
      by (simp add: update.abs_eq eq_onp_same_args)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1170
    finally show ?case
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1171
      by (simp add: Q_def fun_upd_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1172
  qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1173
  then show ?thesis by (simp add: Q_def \<open>f = Abs_poly_mapping g\<close>)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1174
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1175
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1176
lemma lookup_update:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1177
  "lookup (update k v f) k' = (if k = k' then v else lookup f k')"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1178
  by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1179
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1180
lemma keys_update:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1181
  "keys (update k v f) = (if v = 0 then keys f - {k} else insert k (keys f))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1182
  by transfer auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1183
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1184
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1185
subsection \<open>Quasi-functorial structure\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1186
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1187
lift_definition map :: "('b::zero \<Rightarrow> 'c::zero)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1188
  \<Rightarrow> ('a \<Rightarrow>\<^sub>0 'b) \<Rightarrow> ('a \<Rightarrow>\<^sub>0 'c::zero)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1189
  is "\<lambda>g f k. g (f k) when f k \<noteq> 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1190
  by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1191
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1192
context
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1193
  fixes f :: "'b \<Rightarrow> 'a"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1194
  assumes inj_f: "inj f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1195
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1196
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1197
lift_definition map_key :: "('a \<Rightarrow>\<^sub>0 'c::zero) \<Rightarrow> 'b \<Rightarrow>\<^sub>0 'c"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1198
  is "\<lambda>p. p \<circ> f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1199
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1200
  fix g :: "'c \<Rightarrow> 'd" and p :: "'a \<Rightarrow> 'c"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1201
  assume "finite {x. p x \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1202
  hence "finite (f ` {y. p (f y) \<noteq> 0})"
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
  1203
    by (simp add: rev_finite_subset subset_eq)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1204
  thus "finite {x. (p \<circ> f) x \<noteq> 0}" unfolding o_def
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
  1205
    by (metis finite_imageD injD inj_f inj_on_def)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1206
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1207
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1208
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1209
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1210
lemma map_key_compose:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1211
  assumes [transfer_rule]: "inj f" "inj g"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1212
  shows "map_key f (map_key g p) = map_key (g \<circ> f) p"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1213
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1214
  from assms have [transfer_rule]: "inj (g \<circ> f)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1215
    by(simp add: inj_compose)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1216
  show ?thesis by transfer(simp add: o_assoc)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1217
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1218
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1219
lemma map_key_id:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1220
  "map_key (\<lambda>x. x) p = p"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1221
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1222
  have [transfer_rule]: "inj (\<lambda>x. x)" by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1223
  show ?thesis by transfer(simp add: o_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1224
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1225
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1226
context
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1227
  fixes f :: "'a \<Rightarrow> 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1228
  assumes inj_f [transfer_rule]: "inj f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1229
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1230
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1231
lemma map_key_map:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1232
  "map_key f (map g p) = map g (map_key f p)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1233
  by transfer (simp add: fun_eq_iff)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1234
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1235
lemma map_key_plus:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1236
  "map_key f (p + q) = map_key f p + map_key f q"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1237
  by transfer (simp add: fun_eq_iff)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1238
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1239
lemma keys_map_key:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1240
  "keys (map_key f p) = f -` keys p"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1241
  by transfer auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1242
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1243
lemma map_key_zero [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1244
  "map_key f 0 = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1245
  by transfer (simp add: fun_eq_iff)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1246
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1247
lemma map_key_single [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1248
  "map_key f (single (f k) v) = single k v"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1249
  by transfer (simp add: fun_eq_iff inj_onD [OF inj_f] when_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1250
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1251
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1252
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1253
lemma mult_map_scale_conv_mult: "map ((*) s) p = single 0 s * p"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1254
proof(transfer fixing: s)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1255
  fix p :: "'a \<Rightarrow> 'b"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1256
  assume *: "finite {x. p x \<noteq> 0}"
81332
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1257
  have "prod_fun (\<lambda>k'. s when 0 = k') p x = (\<lambda>k. s * p k when p k \<noteq> 0) x" (is "?lhs = ?rhs") for x
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1258
  proof -
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1259
    have "?lhs = (\<Sum>l :: 'a. if l = 0 then s * (\<Sum>q. p q when x = q) else 0)"
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1260
      by (auto simp: prod_fun_def when_def intro: Sum_any.cong simp del: Sum_any.delta)
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1261
    also have "\<dots> = ?rhs"
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1262
      by (simp add: when_def)
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1263
    finally show ?thesis .
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1264
  qed
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1265
  then show "(\<lambda>k. s * p k when p k \<noteq> 0) = prod_fun (\<lambda>k'. s when 0 = k') p"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1266
    by(simp add: fun_eq_iff)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1267
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1268
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1269
lemma map_single [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1270
  "(c = 0 \<Longrightarrow> f 0 = 0) \<Longrightarrow> map f (single x c) = single x (f c)"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1271
  by transfer(auto simp: fun_eq_iff when_def)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1272
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1273
lemma map_eq_zero_iff: "map f p = 0 \<longleftrightarrow> (\<forall>k \<in> keys p. f (lookup p k) = 0)"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1274
  by transfer(auto simp: fun_eq_iff when_def)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1275
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1276
subsection \<open>Canonical dense representation of @{typ "nat \<Rightarrow>\<^sub>0 'a"}\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1277
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1278
abbreviation no_trailing_zeros :: "'a :: zero list \<Rightarrow> bool"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1279
where
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1280
  "no_trailing_zeros \<equiv> no_trailing ((=) 0)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1281
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1282
lift_definition "nth" :: "'a list \<Rightarrow> (nat \<Rightarrow>\<^sub>0 'a::zero)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1283
  is "nth_default 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1284
  by (fact finite_nth_default_neq_default)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1285
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1286
text \<open>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1287
  The opposite direction is directly specified on (later)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1288
  type \<open>nat_mapping\<close>.
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1289
\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1290
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1291
lemma nth_Nil [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1292
  "nth [] = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1293
  by transfer (simp add: fun_eq_iff)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1294
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1295
lemma nth_singleton [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1296
  "nth [v] = single 0 v"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1297
proof (transfer, rule ext)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1298
  fix n :: nat and v :: 'a
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1299
  show "nth_default 0 [v] n = (v when 0 = n)"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1300
    by (auto simp: nth_default_def nth_append)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1301
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1302
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1303
lemma nth_replicate [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1304
  "nth (replicate n 0 @ [v]) = single n v"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1305
proof (transfer, rule ext)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1306
  fix m n :: nat and v :: 'a
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1307
  show "nth_default 0 (replicate n 0 @ [v]) m = (v when n = m)"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1308
    by (auto simp: nth_default_def nth_append)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1309
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1310
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1311
lemma nth_strip_while [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1312
  "nth (strip_while ((=) 0) xs) = nth xs"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1313
  by transfer (fact nth_default_strip_while_dflt)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1314
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1315
lemma nth_strip_while' [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1316
  "nth (strip_while (\<lambda>k. k = 0) xs) = nth xs"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1317
  by (subst eq_commute) (fact nth_strip_while)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1318
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1319
lemma nth_eq_iff:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1320
  "nth xs = nth ys \<longleftrightarrow> strip_while (HOL.eq 0) xs = strip_while (HOL.eq 0) ys"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1321
  by transfer (simp add: nth_default_eq_iff)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1322
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1323
lemma lookup_nth [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1324
  "lookup (nth xs) = nth_default 0 xs"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1325
  by (fact nth.rep_eq)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1326
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1327
lemma keys_nth [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1328
  "keys (nth xs) =  fst ` {(n, v) \<in> set (enumerate 0 xs). v \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1329
proof transfer
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1330
  fix xs :: "'a list"
81332
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1331
  have "n \<in> fst ` {(n, v). (n, v) \<in> set (enumerate 0 xs) \<and> v \<noteq> 0}"
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1332
    if "nth_default 0 xs n \<noteq> 0" for n
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1333
  proof -
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1334
    from that have "n < length xs" and "xs ! n \<noteq> 0"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1335
      by (auto simp: nth_default_def split: if_splits)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1336
    then have "(n, xs ! n) \<in> {(n, v). (n, v) \<in> set (enumerate 0 xs) \<and> v \<noteq> 0}" (is "?x \<in> ?A")
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1337
      by (auto simp: in_set_conv_nth enumerate_eq_zip)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1338
    then have "fst ?x \<in> fst ` ?A"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1339
      by blast
81332
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1340
    then show ?thesis
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1341
      by simp
81332
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1342
  qed
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1343
  then show "{k. nth_default 0 xs k \<noteq> 0} = fst ` {(n, v). (n, v) \<in> set (enumerate 0 xs) \<and> v \<noteq> 0}"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1344
    by (auto simp: in_enumerate_iff_nth_default_eq)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1345
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1346
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1347
lemma range_nth [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1348
  "range (nth xs) = set xs - {0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1349
  by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1350
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1351
lemma degree_nth:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1352
  "no_trailing_zeros xs \<Longrightarrow> degree (nth xs) = length xs"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1353
unfolding degree_def proof transfer
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1354
  fix xs :: "'a list"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1355
  assume *: "no_trailing_zeros xs"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1356
  let ?A = "{n. nth_default 0 xs n \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1357
  let ?f = "nth_default 0 xs"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1358
  let ?bound = "Max (insert 0 (Suc ` {n. ?f n \<noteq> 0}))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1359
  show "?bound = length xs"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1360
  proof (cases "xs = []")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1361
    case False
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1362
    with * obtain n where n: "n < length xs" "xs ! n \<noteq> 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1363
      by (fastforce simp add: no_trailing_unfold last_conv_nth neq_Nil_conv)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1364
    then have "?bound = Max (Suc ` {k. (k < length xs \<longrightarrow> xs ! k \<noteq> (0::'a)) \<and> k < length xs})"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1365
      by (subst Max_insert) (auto simp: nth_default_def)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1366
    also let ?A = "{k. k < length xs \<and> xs ! k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1367
    have "{k. (k < length xs \<longrightarrow> xs ! k \<noteq> (0::'a)) \<and> k < length xs} = ?A" by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1368
    also have "Max (Suc ` ?A) = Suc (Max ?A)" using n
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1369
      by (subst mono_Max_commute [where f = Suc, symmetric]) (auto simp: mono_Suc)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1370
    also {
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1371
      have "Max ?A \<in> ?A" using n Max_in [of ?A] by fastforce
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1372
      hence "Suc (Max ?A) \<le> length xs" by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1373
      moreover from * False have "length xs - 1 \<in> ?A"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1374
        by(auto simp: no_trailing_unfold last_conv_nth)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1375
      hence "length xs - 1 \<le> Max ?A" using Max_ge[of ?A "length xs - 1"] by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1376
      hence "length xs \<le> Suc (Max ?A)" by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1377
      ultimately have "Suc (Max ?A) = length xs" by simp }
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1378
    finally show ?thesis .
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1379
  qed simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1380
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1381
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1382
lemma nth_trailing_zeros [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1383
  "nth (xs @ replicate n 0) = nth xs"
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
  1384
  by (simp add: nth.abs_eq)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1385
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1386
lemma nth_idem:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1387
  "nth (List.map (lookup f) [0..<degree f]) = f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1388
  unfolding degree_def by transfer
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1389
    (auto simp: nth_default_def fun_eq_iff not_less)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1390
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1391
lemma nth_idem_bound:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1392
  assumes "degree f \<le> n"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1393
  shows "nth (List.map (lookup f) [0..<n]) = f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1394
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1395
  from assms obtain m where "n = degree f + m"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1396
    by (blast dest: le_Suc_ex)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1397
  then have "[0..<n] = [0..<degree f] @ [degree f..<degree f + m]"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1398
    by (simp add: upt_add_eq_append [of 0])
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1399
  moreover have "List.map (lookup f) [degree f..<degree f + m] = replicate m 0"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1400
    by (rule replicate_eqI) (auto simp: beyond_degree_lookup_zero)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1401
  ultimately show ?thesis by (simp add: nth_idem)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1402
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1403
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1404
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1405
subsection \<open>Canonical sparse representation of @{typ "'a \<Rightarrow>\<^sub>0 'b"}\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1406
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1407
lift_definition the_value :: "('a \<times> 'b) list \<Rightarrow> 'a \<Rightarrow>\<^sub>0 'b::zero"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1408
  is "\<lambda>xs k. case map_of xs k of None \<Rightarrow> 0 | Some v \<Rightarrow> v"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1409
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1410
  fix xs :: "('a \<times> 'b) list"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1411
  have fin: "finite {k. \<exists>v. map_of xs k = Some v}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1412
    using finite_dom_map_of [of xs] unfolding dom_def by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1413
  then show "finite {k. (case map_of xs k of None \<Rightarrow> 0 | Some v \<Rightarrow> v) \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1414
    using fin by (simp split: option.split)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1415
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1416
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1417
definition items :: "('a::linorder \<Rightarrow>\<^sub>0 'b::zero) \<Rightarrow> ('a \<times> 'b) list"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1418
where
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1419
  "items f = List.map (\<lambda>k. (k, lookup f k)) (sorted_list_of_set (keys f))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1420
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1421
text \<open>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1422
  For the canonical sparse representation we provide both
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1423
  directions of morphisms since the specification of
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1424
  ordered association lists in theory \<open>OAList\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1425
  will support arbitrary linear orders @{class linorder}
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1426
  as keys, not just natural numbers @{typ nat}.
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1427
\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1428
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1429
lemma the_value_items [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1430
  "the_value (items f) = f"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1431
  unfolding items_def
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1432
  by transfer (simp add: fun_eq_iff map_of_map_restrict restrict_map_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1433
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1434
lemma lookup_the_value:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1435
  "lookup (the_value xs) k = (case map_of xs k of None \<Rightarrow> 0 | Some v \<Rightarrow> v)"
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
  1436
  by (simp add: the_value.rep_eq)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1437
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1438
lemma items_the_value:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1439
  assumes "sorted (List.map fst xs)" and "distinct (List.map fst xs)" and "0 \<notin> snd ` set xs"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1440
  shows "items (the_value xs) = xs"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1441
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1442
  from assms have "sorted_list_of_set (set (List.map fst xs)) = List.map fst xs"
76484
defaa0b17424 generalized sorted_sort_id to sort_key_id_if_sorted
nipkow
parents: 75455
diff changeset
  1443
    unfolding sorted_list_of_set_sort_remdups by (simp add: distinct_remdups_id sort_key_id_if_sorted)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1444
  moreover from assms have "keys (the_value xs) = fst ` set xs"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1445
    by transfer (auto simp: image_def split: option.split dest: set_map_of_compr)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1446
  ultimately show ?thesis
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1447
    unfolding items_def using assms
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1448
    by (auto simp: lookup_the_value intro: map_idI)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1449
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1450
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1451
lemma the_value_Nil [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1452
  "the_value [] = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1453
  by transfer (simp add: fun_eq_iff)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1454
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1455
lemma the_value_Cons [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1456
  "the_value (x # xs) = update (fst x) (snd x) (the_value xs)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1457
  by transfer (simp add: fun_eq_iff)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1458
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1459
lemma items_zero [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1460
  "items 0 = []"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1461
  unfolding items_def by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1462
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1463
lemma items_one [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1464
  "items 1 = [(0, 1)]"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1465
  unfolding items_def by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1466
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1467
lemma items_single [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1468
  "items (single k v) = (if v = 0 then [] else [(k, v)])"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1469
  unfolding items_def by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1470
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1471
lemma in_set_items_iff [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1472
  "(k, v) \<in> set (items f) \<longleftrightarrow> k \<in> keys f \<and> lookup f k = v"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1473
  unfolding items_def by transfer auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1474
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1475
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1476
subsection \<open>Size estimation\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1477
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1478
context
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1479
  fixes f :: "'a \<Rightarrow> nat"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1480
  and g :: "'b :: zero \<Rightarrow> nat"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1481
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1482
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1483
definition poly_mapping_size :: "('a \<Rightarrow>\<^sub>0 'b) \<Rightarrow> nat"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1484
where
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1485
  "poly_mapping_size m = g 0 + (\<Sum>k \<in> keys m. Suc (f k + g (lookup m k)))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1486
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1487
lemma poly_mapping_size_0 [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1488
  "poly_mapping_size 0 = g 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1489
  by (simp add: poly_mapping_size_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1490
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1491
lemma poly_mapping_size_single [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1492
  "poly_mapping_size (single k v) = (if v = 0 then g 0 else g 0 + f k + g v + 1)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1493
  unfolding poly_mapping_size_def by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1494
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1495
lemma keys_less_poly_mapping_size:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1496
  "k \<in> keys m \<Longrightarrow> f k + g (lookup m k) < poly_mapping_size m"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1497
unfolding poly_mapping_size_def
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1498
proof transfer
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1499
  fix k :: 'a and m :: "'a \<Rightarrow> 'b" and f :: "'a \<Rightarrow> nat" and g
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1500
  let ?keys = "{k. m k \<noteq> 0}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1501
  assume *: "finite ?keys" "k \<in> ?keys"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1502
  then have "f k + g (m k) = (\<Sum>k' \<in> ?keys. f k' + g (m k') when k' = k)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1503
    by (simp add: sum.delta when_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1504
  also have "\<dots> < (\<Sum>k' \<in> ?keys. Suc (f k' + g (m k')))" using *
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1505
    by (intro sum_strict_mono) (auto simp: when_def)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1506
  also have "\<dots> \<le> g 0 + \<dots>" by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1507
  finally have "f k + g (m k) < \<dots>" .
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1508
  then show "f k + g (m k) < g 0 + (\<Sum>k | m k \<noteq> 0. Suc (f k + g (m k)))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1509
    by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1510
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1511
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1512
lemma lookup_le_poly_mapping_size:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1513
  "g (lookup m k) \<le> poly_mapping_size m"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1514
proof (cases "k \<in> keys m")
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1515
  case True
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1516
  with keys_less_poly_mapping_size [of k m]
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1517
  show ?thesis by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1518
next
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1519
  case False
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1520
  then show ?thesis
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1521
    by (simp add: Poly_Mapping.poly_mapping_size_def in_keys_iff)
81332
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1522
qed
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1523
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1524
lemma poly_mapping_size_estimation:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1525
  "k \<in> keys m \<Longrightarrow> y \<le> f k + g (lookup m k) \<Longrightarrow> y < poly_mapping_size m"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1526
  using keys_less_poly_mapping_size by (auto intro: le_less_trans)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1527
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1528
lemma poly_mapping_size_estimation2:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1529
  assumes "v \<in> range m" and "y \<le> g v"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1530
  shows "y < poly_mapping_size m"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1531
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1532
  from assms obtain k where *: "lookup m k = v" "v \<noteq> 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1533
    by transfer blast
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
  1534
  then have "k \<in> keys m"
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1535
    by (simp add: in_keys_iff)
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
  1536
  with * show ?thesis
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
  1537
    by (simp add: Poly_Mapping.poly_mapping_size_estimation assms(2) trans_le_add2)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1538
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1539
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1540
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1541
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1542
lemma poly_mapping_size_one [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1543
  "poly_mapping_size f g 1 = g 0 + f 0 + g 1 + 1"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1544
  unfolding poly_mapping_size_def by transfer simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1545
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1546
lemma poly_mapping_size_cong [fundef_cong]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1547
  "m = m' \<Longrightarrow> g 0 = g' 0 \<Longrightarrow> (\<And>k. k \<in> keys m' \<Longrightarrow> f k = f' k)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1548
    \<Longrightarrow> (\<And>v. v \<in> range m' \<Longrightarrow> g v = g' v)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1549
    \<Longrightarrow> poly_mapping_size f g m = poly_mapping_size f' g' m'"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1550
  by (auto simp: poly_mapping_size_def intro!: sum.cong)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1551
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1552
instantiation poly_mapping :: (type, zero) size
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1553
begin
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1554
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1555
definition "size = poly_mapping_size (\<lambda>_. 0) (\<lambda>_. 0)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1556
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1557
instance ..
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1558
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1559
end
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1560
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1561
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1562
subsection \<open>Further mapping operations and properties\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1563
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1564
text \<open>It is like in algebra: there are many definitions, some are also used\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1565
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1566
lift_definition mapp ::
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1567
  "('a \<Rightarrow> 'b :: zero \<Rightarrow> 'c :: zero) \<Rightarrow> ('a \<Rightarrow>\<^sub>0 'b) \<Rightarrow> ('a \<Rightarrow>\<^sub>0 'c)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1568
  is "\<lambda>f p k. (if k \<in> keys p then f k (lookup p k) else 0)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1569
  by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1570
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1571
lemma mapp_cong [fundef_cong]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1572
  "\<lbrakk> m = m'; \<And>k. k \<in> keys m' \<Longrightarrow> f k (lookup m' k) = f' k (lookup m' k) \<rbrakk>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1573
  \<Longrightarrow> mapp f m = mapp f' m'"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1574
  by transfer (auto simp: fun_eq_iff)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1575
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1576
lemma lookup_mapp:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1577
  "lookup (mapp f p) k = (f k (lookup p k) when k \<in> keys p)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1578
  by (simp add: mapp.rep_eq)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1579
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1580
lemma keys_mapp_subset: "keys (mapp f p) \<subseteq> keys p"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1581
  by (meson in_keys_iff mapp.rep_eq subsetI)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1582
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1583
subsection\<open>Free Abelian Groups Over a Type\<close>
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1584
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1585
abbreviation frag_of :: "'a \<Rightarrow> 'a \<Rightarrow>\<^sub>0 int"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1586
  where "frag_of c \<equiv> Poly_Mapping.single c (1::int)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1587
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1588
lemma lookup_frag_of [simp]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1589
   "Poly_Mapping.lookup(frag_of c) = (\<lambda>x. if x = c then 1 else 0)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1590
  by (force simp add: lookup_single_not_eq)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1591
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1592
lemma frag_of_nonzero [simp]: "frag_of a \<noteq> 0"
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
  1593
  by (metis lookup_single_eq lookup_zero zero_neq_one)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1594
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1595
definition frag_cmul :: "int \<Rightarrow> ('a \<Rightarrow>\<^sub>0 int) \<Rightarrow> ('a \<Rightarrow>\<^sub>0 int)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1596
  where "frag_cmul c a = Abs_poly_mapping (\<lambda>x. c * Poly_Mapping.lookup a x)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1597
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1598
lemma frag_cmul_zero [simp]: "frag_cmul 0 x = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1599
  by (simp add: frag_cmul_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1600
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1601
lemma frag_cmul_zero2 [simp]: "frag_cmul c 0 = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1602
  by (simp add: frag_cmul_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1603
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1604
lemma frag_cmul_one [simp]: "frag_cmul 1 x = x"
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
  1605
  by (simp add: frag_cmul_def)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1606
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1607
lemma frag_cmul_minus_one [simp]: "frag_cmul (-1) x = -x"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1608
  by (simp add: frag_cmul_def uminus_poly_mapping_def poly_mapping_eqI)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1609
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1610
lemma frag_cmul_cmul [simp]: "frag_cmul c (frag_cmul d x) = frag_cmul (c*d) x"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1611
  by (simp add: frag_cmul_def mult_ac)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1612
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1613
lemma lookup_frag_cmul [simp]: "poly_mapping.lookup (frag_cmul c x) i = c * poly_mapping.lookup x i"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1614
  by (simp add: frag_cmul_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1615
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1616
lemma minus_frag_cmul [simp]: "- frag_cmul k x = frag_cmul (-k) x"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1617
  by (simp add: poly_mapping_eqI)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1618
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1619
lemma keys_frag_of: "Poly_Mapping.keys(frag_of a) = {a}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1620
  by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1621
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1622
lemma finite_cmul_nonzero: "finite {x. c * Poly_Mapping.lookup a x \<noteq> (0::int)}"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1623
  by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1624
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1625
lemma keys_cmul: "Poly_Mapping.keys(frag_cmul c a) \<subseteq> Poly_Mapping.keys a"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1626
  using finite_cmul_nonzero [of c a]
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1627
  by (metis lookup_frag_cmul mult_zero_right not_in_keys_iff_lookup_eq_zero subsetI)
81332
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1628
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1629
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1630
lemma keys_cmul_iff [iff]: "i \<in> Poly_Mapping.keys (frag_cmul c x) \<longleftrightarrow> i \<in> Poly_Mapping.keys x \<and> c \<noteq> 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1631
  by (metis in_keys_iff lookup_frag_cmul mult_eq_0_iff)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1632
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1633
lemma keys_minus [simp]: "Poly_Mapping.keys(-a) = Poly_Mapping.keys a"
73932
fd21b4a93043 added opaque_combs and renamed hide_lams to opaque_lifting
desharna
parents: 73655
diff changeset
  1634
  by (metis (no_types, opaque_lifting) in_keys_iff lookup_uminus neg_equal_0_iff_equal subsetI subset_antisym)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1635
81332
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1636
lemma keys_diff:
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1637
  "Poly_Mapping.keys(a - b) \<subseteq> Poly_Mapping.keys a \<union> Poly_Mapping.keys b"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1638
  by (auto simp: in_keys_iff lookup_minus)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1639
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1640
lemma keys_eq_empty [simp]: "Poly_Mapping.keys c = {} \<longleftrightarrow> c = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1641
  by (metis in_keys_iff keys_zero lookup_zero poly_mapping_eqI)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1642
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1643
lemma frag_cmul_eq_0_iff [simp]: "frag_cmul k c = 0 \<longleftrightarrow> k=0 \<or> c=0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1644
  by auto (metis subsetI subset_antisym keys_cmul_iff keys_eq_empty)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1645
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1646
lemma frag_of_eq: "frag_of x = frag_of y \<longleftrightarrow> x = y"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1647
  by (metis lookup_single_eq lookup_single_not_eq zero_neq_one)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1648
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1649
lemma frag_cmul_distrib: "frag_cmul (c+d) a = frag_cmul c a + frag_cmul d a"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1650
  by (simp add: frag_cmul_def plus_poly_mapping_def int_distrib)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1651
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1652
lemma frag_cmul_distrib2: "frag_cmul c (a+b) = frag_cmul c a + frag_cmul c b"
81816
bee084ecd18c More tidying of old proofs
paulson <lp15@cam.ac.uk>
parents: 81332
diff changeset
  1653
  by (simp add: int_distrib(2) lookup_add poly_mapping_eqI)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1654
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1655
lemma frag_cmul_diff_distrib: "frag_cmul (a - b) c = frag_cmul a c - frag_cmul b c"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1656
  by (auto simp: left_diff_distrib lookup_minus poly_mapping_eqI)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1657
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1658
lemma frag_cmul_sum:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1659
     "frag_cmul a (sum b I) = (\<Sum>i\<in>I. frag_cmul a (b i))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1660
proof (induction rule: infinite_finite_induct)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1661
  case (insert i I)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1662
  then show ?case
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1663
    by (auto simp: algebra_simps frag_cmul_distrib2)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1664
qed auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1665
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1666
lemma keys_sum: "Poly_Mapping.keys(sum b I) \<subseteq> (\<Union>i \<in>I. Poly_Mapping.keys(b i))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1667
proof (induction I rule: infinite_finite_induct)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1668
  case (insert i I)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1669
  then show ?case
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1670
    using keys_add [of "b i" "sum b I"] by auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1671
qed auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1672
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1673
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1674
definition frag_extend :: "('b \<Rightarrow> 'a \<Rightarrow>\<^sub>0 int) \<Rightarrow> ('b \<Rightarrow>\<^sub>0 int) \<Rightarrow> 'a \<Rightarrow>\<^sub>0 int"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1675
  where "frag_extend b x \<equiv> (\<Sum>i \<in> Poly_Mapping.keys x. frag_cmul (Poly_Mapping.lookup x i) (b i))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1676
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1677
lemma frag_extend_0 [simp]: "frag_extend b 0 = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1678
  by (simp add: frag_extend_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1679
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1680
lemma frag_extend_of [simp]: "frag_extend f (frag_of a) = f a"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1681
  by (simp add: frag_extend_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1682
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1683
lemma frag_extend_cmul:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1684
   "frag_extend f (frag_cmul c x) = frag_cmul c (frag_extend f x)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1685
  by (auto simp: frag_extend_def frag_cmul_sum intro: sum.mono_neutral_cong_left)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1686
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1687
lemma frag_extend_minus:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1688
   "frag_extend f (- x) = - (frag_extend f x)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1689
  using frag_extend_cmul [of f "-1"] by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1690
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1691
lemma frag_extend_add:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1692
  "frag_extend f (a+b) = (frag_extend f a) + (frag_extend f b)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1693
proof -
81332
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1694
  have *: "(\<Sum>i\<in>Poly_Mapping.keys a. frag_cmul (poly_mapping.lookup a i) (f i))
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1695
         = (\<Sum>i\<in>Poly_Mapping.keys a \<union> Poly_Mapping.keys b. frag_cmul (poly_mapping.lookup a i) (f i))"
81332
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1696
          "(\<Sum>i\<in>Poly_Mapping.keys b. frag_cmul (poly_mapping.lookup b i) (f i))
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1697
         = (\<Sum>i\<in>Poly_Mapping.keys a \<union> Poly_Mapping.keys b. frag_cmul (poly_mapping.lookup b i) (f i))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1698
    by (auto simp: in_keys_iff intro: sum.mono_neutral_cong_left)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1699
  have "frag_extend f (a+b) = (\<Sum>i\<in>Poly_Mapping.keys (a + b).
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1700
       frag_cmul (poly_mapping.lookup a i) (f i) + frag_cmul (poly_mapping.lookup b i) (f i)) "
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1701
    by (auto simp: frag_extend_def Poly_Mapping.lookup_add frag_cmul_distrib)
81332
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1702
  also have "... = (\<Sum>i \<in> Poly_Mapping.keys a \<union> Poly_Mapping.keys b. frag_cmul (poly_mapping.lookup a i) (f i)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1703
                         + frag_cmul (poly_mapping.lookup b i) (f i))"
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1704
  proof (rule sum.mono_neutral_cong_left)
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1705
    show "\<forall>i\<in>keys a \<union> keys b - keys (a + b).
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1706
       frag_cmul (lookup a i) (f i) + frag_cmul (lookup b i) (f i) = 0"
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1707
      by (metis DiffD2 frag_cmul_distrib frag_cmul_zero in_keys_iff lookup_add)
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1708
  qed (auto simp: keys_add)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1709
  also have "... = (frag_extend f a) + (frag_extend f b)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1710
    by (auto simp: * sum.distrib frag_extend_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1711
  finally show ?thesis .
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1712
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1713
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1714
lemma frag_extend_diff:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1715
   "frag_extend f (a-b) = (frag_extend f a) - (frag_extend f b)"
73932
fd21b4a93043 added opaque_combs and renamed hide_lams to opaque_lifting
desharna
parents: 73655
diff changeset
  1716
  by (metis (no_types, opaque_lifting) add_uminus_conv_diff frag_extend_add frag_extend_minus)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1717
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1718
lemma frag_extend_sum:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1719
   "finite I \<Longrightarrow> frag_extend f (\<Sum>i\<in>I. g i) = sum (frag_extend f o g) I"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1720
  by (induction I rule: finite_induct) (simp_all add: frag_extend_add)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1721
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1722
lemma frag_extend_eq:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1723
   "(\<And>f. f \<in> Poly_Mapping.keys c \<Longrightarrow> g f = h f) \<Longrightarrow> frag_extend g c = frag_extend h c"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1724
  by (simp add: frag_extend_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1725
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1726
lemma frag_extend_eq_0:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1727
   "(\<And>x. x \<in> Poly_Mapping.keys c \<Longrightarrow> f x = 0) \<Longrightarrow> frag_extend f c = 0"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1728
  by (simp add: frag_extend_def)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1729
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1730
lemma keys_frag_extend: "Poly_Mapping.keys(frag_extend f c) \<subseteq> (\<Union>x \<in> Poly_Mapping.keys c. Poly_Mapping.keys(f x))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1731
  unfolding frag_extend_def
73655
26a1d66b9077 tuned proofs --- avoid z3, which is absent on arm64-linux;
wenzelm
parents: 70045
diff changeset
  1732
  using keys_sum by fastforce
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1733
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1734
lemma frag_expansion: "a = frag_extend frag_of a"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1735
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1736
  have *: "finite I
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1737
        \<Longrightarrow> Poly_Mapping.lookup (\<Sum>i\<in>I. frag_cmul (Poly_Mapping.lookup a i) (frag_of i)) j =
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1738
            (if j \<in> I then Poly_Mapping.lookup a j else 0)" for I j
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1739
    by (induction I rule: finite_induct) (auto simp: lookup_single lookup_add)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1740
  show ?thesis
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1741
    unfolding frag_extend_def
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1742
    by (rule poly_mapping_eqI) (fastforce simp add: in_keys_iff *)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1743
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1744
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1745
lemma frag_closure_minus_cmul:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1746
  assumes "P 0" and P: "\<And>x y. \<lbrakk>P x; P y\<rbrakk> \<Longrightarrow> P(x - y)" "P c"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1747
  shows "P(frag_cmul k c)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1748
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1749
  have "P (frag_cmul (int n) c)" for n
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1750
  proof (induction n)
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1751
    case 0
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1752
    then show ?case
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1753
      by (simp add: assms)
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1754
  next
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1755
    case (Suc n)
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1756
    then show ?case
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1757
      by (metis assms diff_0 diff_minus_eq_add frag_cmul_distrib frag_cmul_one of_nat_Suc)
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1758
  qed
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1759
  then show ?thesis
73932
fd21b4a93043 added opaque_combs and renamed hide_lams to opaque_lifting
desharna
parents: 73655
diff changeset
  1760
    by (metis (no_types, opaque_lifting) add_diff_eq assms(2) diff_add_cancel frag_cmul_distrib int_diff_cases)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1761
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1762
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1763
lemma frag_induction [consumes 1, case_names zero one diff]:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1764
  assumes supp: "Poly_Mapping.keys c \<subseteq> S"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1765
    and 0: "P 0" and sing: "\<And>x. x \<in> S \<Longrightarrow> P(frag_of x)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1766
    and diff: "\<And>a b. \<lbrakk>P a; P b\<rbrakk> \<Longrightarrow> P(a - b)"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1767
  shows "P c"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1768
proof -
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1769
  have "P (\<Sum>i\<in>I. frag_cmul (poly_mapping.lookup c i) (frag_of i))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1770
    if "I \<subseteq> Poly_Mapping.keys c" for I
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1771
    using finite_subset [OF that finite_keys [of c]] that supp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1772
  proof (induction I arbitrary: c rule: finite_induct)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1773
    case empty
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1774
    then show ?case
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1775
      by (auto simp: 0)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1776
  next
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1777
    case (insert i I c)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1778
    have ab: "a+b = a - (0 - b)" for a b :: "'a \<Rightarrow>\<^sub>0 int"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1779
      by simp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1780
    have Pfrag: "P (frag_cmul (poly_mapping.lookup c i) (frag_of i))"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1781
      by (metis "0" diff frag_closure_minus_cmul insert.prems insert_subset sing subset_iff)
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1782
    with insert show ?case
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1783
      by (metis (mono_tags, lifting) "0" ab diff insert_subset sum.insert)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1784
  qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1785
  then show ?thesis
80095
0f9cd1a5edbe Tidying ugly proofs
paulson <lp15@cam.ac.uk>
parents: 77955
diff changeset
  1786
    by (subst frag_expansion) (auto simp: frag_extend_def)
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1787
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1788
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1789
lemma frag_extend_compose:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1790
  "frag_extend f (frag_extend (frag_of o g) c) = frag_extend (f o g) c"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1791
  using subset_UNIV
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1792
  by (induction c rule: frag_induction) (auto simp: frag_extend_diff)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1793
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1794
lemma frag_split:
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1795
  fixes c :: "'a \<Rightarrow>\<^sub>0 int"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1796
  assumes "Poly_Mapping.keys c \<subseteq> S \<union> T"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1797
  obtains d e where "Poly_Mapping.keys d \<subseteq> S" "Poly_Mapping.keys e \<subseteq> T" "d + e = c"
81332
f94b30fa2b6c tuned proofs;
wenzelm
parents: 80914
diff changeset
  1798
proof
70043
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1799
  let ?d = "frag_extend (\<lambda>f. if f \<in> S then frag_of f else 0) c"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1800
  let ?e = "frag_extend (\<lambda>f. if f \<in> S then 0 else frag_of f) c"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1801
  show "Poly_Mapping.keys ?d \<subseteq> S" "Poly_Mapping.keys ?e \<subseteq> T"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1802
    using assms by (auto intro!: order_trans [OF keys_frag_extend] split: if_split_asm)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1803
  show "?d + ?e = c"
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1804
    using assms
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1805
  proof (induction c rule: frag_induction)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1806
    case (diff a b)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1807
    then show ?case
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1808
      by (metis (no_types, lifting) frag_extend_diff add_diff_eq diff_add_eq diff_add_eq_diff_diff_swap)
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1809
  qed auto
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1810
qed
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1811
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1812
hide_const (open) lookup single update keys range map map_key degree nth the_value items foldr mapp
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1813
350acd367028 theory Poly_Mapping
paulson <lp15@cam.ac.uk>
parents:
diff changeset
  1814
end