lemma coeff_list: "map (coeff p) (rev [0..< length p]) = p" proof (induction p) case Nil thus ?caseby simp next case (Cons a p) have"map (coeff (a # p)) (rev [0.. a # (map (coeff p) (rev [0.. by auto alsohave" ... = a # p" using Cons by simp finallyshow ?case . qed
lemma coeff_nth: "i < length p ==> (coeff p) i = p ! (length p - 1 - i)" proof - assume i_lt: "i < length p" hence"(coeff p) i = (map (coeff p) [0..< length p]) ! i" by simp alsohave" ... = (rev (map (coeff p) (rev [0..< length p]))) ! i" by (simp add: rev_map) alsohave" ... = (map (coeff p) (rev [0..< length p])) ! (length p - 1 - i)" using coeff_list i_lt rev_nth by auto alsohave" ... = p ! (length p - 1 - i)" using coeff_list[of p] by simp finallyshow"(coeff p) i = p ! (length p - 1 - i)" . qed
lemma coeff_iff_length_cond: assumes"length p1 = length p2" shows"p1 = p2 ⟷ coeff p1 = coeff p2" proof show"p1 = p2 ==> coeff p1 = coeff p2" by simp next assume A: "coeff p1 = coeff p2" have"p1 = map (coeff p1) (rev [0..< length p1])" using coeff_list[of p1] by simp alsohave" ... = map (coeff p2) (rev [0..< length p2])" using A assms by simp alsohave" ... = p2" using coeff_list[of p2] by simp finallyshow"p1 = p2" . qed
lemma coeff_img_restrict: "(coeff p) ` {..< length p} = set p" using coeff_list[of p] by (metis atLeast_upt image_set set_rev)
lemma coeff_length: "∧i. i ≥ length p ==> (coeff p) i = 0" by (induction p) (auto)
lemma coeff_degree: "∧i. i > degree p ==> (coeff p) i = 0" using coeff_length by (simp)
lemma replicate_zero_coeff [simp]: "coeff (replicate n 0) = (λ_. 0)" by (induction n) (auto)
lemma scalar_coeff: "a ∈ carrier R ==> coeff (map (λb. a ⊗ b) p) = (λi. a ⊗ (coeff p) i)" by (induction p) (auto)
lemma monom_coeff: "coeff (monom a n) = (λi. if i = n then a else 0)" unfolding monom_def by (induction n) (auto)
lemma coeff_img: "(coeff p) ` {..< length p} = set p" "(coeff p) ` { length p ..} = { 0 }" "(coeff p) ` UNIV = (set p) ∪ { 0 }" using coeff_img_restrict proof (simp) show coeff_img_up: "(coeff p) ` { length p ..} = { 0 }" using coeff_length[of p] by force from coeff_img_up and coeff_img_restrict[of p] show"(coeff p) ` UNIV = (set p) ∪ { 0 }" by force qed
lemma degree_def': assumes"polynomial K p" shows"degree p = (LEAST n. ∀i. i > n ⟶ (coeff p) i = 0)" proof (cases p) case Nil thus ?thesis by auto next
define P where"P = (λn. ∀i. i > n ⟶ (coeff p) i = 0)"
case (Cons a ps) hence"(coeff p) (degree p) ≠0" using assms unfolding polynomial_def by auto hence"∧n. n < degree p ==>¬ P n" unfolding P_def by auto moreoverhave"P (degree p)" unfolding P_def using coeff_degree[of p] by simp ultimatelyhave"degree p = (LEAST n. P n)" by (meson LeastI nat_neq_iff not_less_Least) thus ?thesis unfolding P_def . qed
lemma coeff_iff_polynomial_cond: assumes"polynomial K p1"and"polynomial K p2" shows"p1 = p2 ⟷ coeff p1 = coeff p2" proof show"p1 = p2 ==> coeff p1 = coeff p2" by simp next assume coeff_eq: "coeff p1 = coeff p2" hence deg_eq: "degree p1 = degree p2" using degree_def'[OF assms(1)] degree_def'[OF assms(2)] by auto thus"p1 = p2" proof (cases "p1 ≠ [] ∧ p2 ≠ []") case True hence"length p1 = length p2" using deg_eq by (simp add: Nitpick.size_list_simp(2)) thus ?thesis using coeff_iff_length_cond[of p1 p2] coeff_eq by simp next case False have aux_lemma: "p2 = []" if A: "p1 = []""coeff p1 = coeff p2""polynomial K p2" for p1 p2 proof (rule ccontr) assume"p2 ≠ []" hence"(coeff p2) (degree p2) ≠0" using A(3) unfolding polynomial_def by (metis coeff.simps(2) list.collapse) moreoverhave"(coeff p1) ` UNIV = { 0 }" using A(1) by auto hence"(coeff p2) ` UNIV = { 0 }" using A(2) by simp ultimatelyshow False by blast qed from False have"p1 = [] ∨ p2 = []"by simp thus ?thesis using assms coeff_eq aux_lemma[of p1 p2] aux_lemma[of p2 p1] by auto qed qed
lemma normalize_lead_coeff: assumes"length (normalize p) < length p" shows"lead_coeff p = 0" proof (cases p) case Nil thus ?thesis using assms by simp next case (Cons a ps) thus ?thesis using assms by (cases "a = 0") (auto) qed
lemma normalize_length_lt: assumes"lead_coeff p = 0"and"length p > 0" shows"length (normalize p) < length p" proof (cases p) case Nil thus ?thesis using assms by simp next case (Cons a ps) thus ?thesis using normalize_length_le[of ps] assms by simp qed
lemma normalize_length_eq: assumes"lead_coeff p ≠0" shows"length (normalize p) = length p" using normalize_length_le[of p] assms nat_less_le normalize_lead_coeff by auto
lemma normalize_replicate_zero: "normalize ((replicate n 0) @ p) = normalize p" by (induction n) (auto)
lemma normalize_def': shows"p = (replicate (length p - length (normalize p)) 0) @ (drop (length p - length (normalize p)) p)" (is ?statement1) and"normalize p = drop (length p - length (normalize p)) p" (is ?statement2) proof - show ?statement1 proof (induction p) case Nil thus ?caseby simp next case (Cons a p) thus ?case proof (cases "a = 0") assume"a ≠0"thus ?case using Cons by simp next assume eq_zero: "a = 0" hence len_eq: "Suc (length p - length (normalize p)) = length (a # p) - length (normalize (a # p))" by (simp add: Suc_diff_le normalize_length_le) have"a # p = 0 # (replicate (length p - length (normalize p)) 0 @ drop (length p - length (normalize p)) p)" using eq_zero Cons by simp alsohave" ... = (replicate (Suc (length p - length (normalize p))) 0 @ drop (Suc (length p - length (normalize p))) (a # p))" by simp alsohave" ... = (replicate (length (a # p) - length (normalize (a # p))) 0 @ drop (length (a # p) - length (normalize (a # p))) (a # p))" using len_eq by simp finallyshow ?case . qed qed next show ?statement2 proof - have"∃m. normalize p = drop m p" proof (induction p) case Nil thus ?caseby simp next case (Cons a p) thus ?case apply (cases "a = 0") apply (auto) apply (metis drop_Suc_Cons) apply (metis drop0) done qed thenobtain m where m: "normalize p = drop m p"by auto hence"length (normalize p) = length p - m"by simp thus ?thesis using m by (metis rev_drop rev_rev_ident take_rev) qed qed
lemma normalize_coeff: "coeff p = coeff (normalize p)" proof (induction p) case Nil thus ?caseby simp next case (Cons a p) have"coeff (normalize p) (length p) = 0" using normalize_length_le[of p] coeff_degree[of "normalize p"] coeff_length by blast thenshow ?case using Cons by (cases "a = 0") (auto) qed
lemma append_coeff: "coeff (p @ q) = (λi. if i < length q then (coeff q) i else (coeff p) (i - length q))" proof (induction p) case Nil thus ?case using coeff_length[of q] by auto next case (Cons a p) have"coeff ((a # p) @ q) = (λi. if i = length p + length q then a else (coeff (p @ q)) i)" by auto alsohave" ... = (λi. if i = length p + length q then a else if i < length q then (coeff q) i else (coeff p) (i - length q))" using Cons by auto alsohave" ... = (λi. if i < length q then (coeff q) i else if i = length p + length q then a else (coeff p) (i - length q))" by auto alsohave" ... = (λi. if i < length q then (coeff q) i else if i - length q = length p then a else (coeff p) (i - length q))" by fastforce alsohave" ... = (λi. if i < length q then (coeff q) i else (coeff (a # p)) (i - length q))" by auto finallyshow ?case . qed
lemma prefix_replicate_zero_coeff: "coeff p = coeff ((replicate n 0) @ p)" using append_coeff[of "replicate n 0" p] replicate_zero_coeff[of n] coeff_length[of p] by auto
(* ========================================================================== *) context fixes K :: "'a set"assumes K: "subring K R" begin
lemma polynomial_in_carrier [intro]: "polynomial K p ==> set p ⊆ carrier R" unfolding polynomial_def using subringE(1)[OF K] by auto
lemma carrier_polynomial [intro]: "polynomial K p ==> polynomial (carrier R) p" unfolding polynomial_def using subringE(1)[OF K] by auto
lemma append_is_polynomial: "[ polynomial K p; p ≠ [] ]==> polynomial K (p @ (replicate n 0))" unfolding polynomial_def using subringE(2)[OF K] by auto
lemma lead_coeff_in_carrier: "polynomial K (a # p) ==> a ∈ carrier R - { 0 }" unfolding polynomial_def using subringE(1)[OF K] by auto
lemma monom_is_polynomial [intro]: "a ∈ K - { 0 } ==> polynomial K (monom a n)" unfolding polynomial_def monom_def using subringE(2)[OF K] by auto
lemma eval_poly_in_carrier: "[ polynomial K p; x ∈ carrier R ]==> (eval p) x ∈ carrier R" using eval_in_carrier[OF polynomial_in_carrier] .
lemma poly_coeff_in_carrier [simp]: "polynomial K p ==> coeff p i ∈ carrier R" using coeff_in_carrier[OF polynomial_in_carrier] .
end(* of fixed K context. *) (* ========================================================================== *)
subsection‹Polynomial Addition›
(* ========================================================================== *) context fixes K :: "'a set"assumes K: "subring K R" begin
lemma poly_add_is_polynomial: assumes"set p1 ⊆ K"and"set p2 ⊆ K" shows"polynomial K (poly_add p1 p2)" proof - have"polynomial K (poly_add p1 p2)" if A: "set p1 ⊆ K""set p2 ⊆ K""length p1 ≥ length p2"for p1 p2 proof -
define p2' where"p2' = (replicate (length p1 - length p2) 0) @ p2" hence"set p2' ⊆ K"and"length p1 = length p2'" using A(2-3) subringE(2)[OF K] by auto hence"set (map2 (⊕) p1 p2') ⊆ K" using A(1) subringE(7)[OF K] by (induct p1) (auto, metis set_ConsD subsetD set_zip_leftD set_zip_rightD) thus ?thesis unfolding p2'_defusing normalize_gives_polynomial A(3) by simp qed thus ?thesis using assms by auto qed
lemma poly_add_closed: "[ polynomial K p1; polynomial K p2 ]==> polynomial K (poly_add p1 p2)" using poly_add_is_polynomial polynomial_incl by simp
lemma poly_add_length_eq: assumes"polynomial K p1""polynomial K p2"and"length p1 ≠ length p2" shows"length (poly_add p1 p2) = max (length p1) (length p2)" proof - have"length (poly_add p1 p2) = max (length p1) (length p2)" if A: "polynomial K p1""polynomial K p2""length p1 > length p2"for p1 p2 proof - let ?p2 = "(replicate (length p1 - length p2) 0) @ p2" have p1: "p1 ≠ []"and p2: "?p2 ≠ []" using A(3) by auto thenhave"zip p1 (replicate (length p1 - length p2) 0 @ p2) = zip (lead_coeff p1 # tl p1) (lead_coeff (replicate (length p1 - length p2) 0 @ p2) # tl (replicate (length p1 - length p2) 0 @ p2))" by auto hence"lead_coeff (map2 (⊕) p1 ?p2) = lead_coeff p1 ⊕ lead_coeff ?p2" by simp moreoverhave"lead_coeff p1 ∈ carrier R" using p1 A(1) lead_coeff_in_carrier[OF K, of "hd p1""tl p1"] by auto ultimatelyhave"lead_coeff (map2 (⊕) p1 ?p2) = lead_coeff p1" using A(3) by auto moreoverhave"lead_coeff p1 ≠0" using p1 A(1) unfolding polynomial_def by simp ultimatelyhave"length (normalize (map2 (⊕) p1 ?p2)) = length p1" using normalize_length_eq by auto thus ?thesis using A(3) by auto qed thus ?thesis using assms by auto qed
lemma poly_add_degree_eq: assumes"polynomial K p1""polynomial K p2"and"degree p1 ≠ degree p2" shows"degree (poly_add p1 p2) = max (degree p1) (degree p2)" using poly_add_length_eq[OF assms(1-2)] assms(3) by simp
end(* of fixed K context. *) (* ========================================================================== *)
lemma poly_add_in_carrier: "[ set p1 ⊆ carrier R; set p2 ⊆ carrier R ]==> set (poly_add p1 p2) ⊆ carrier R" using polynomial_incl[OF poly_add_is_polynomial[OF carrier_is_subring]] by simp
lemma poly_add_length_le: "length (poly_add p1 p2) ≤ max (length p1) (length p2)" proof - have"length (poly_add p1 p2) ≤ max (length p1) (length p2)" if"length p1 ≥ length p2"for p1 p2 :: "'a list" using normalize_length_le[of "map2 (⊕) p1 ((replicate (length p1 - length p2) 0) @ p2)"] that by auto thus ?thesis by (metis le_cases max.commute poly_add.simps) qed
lemma poly_add_degree: "degree (poly_add p1 p2) ≤ max (degree p1) (degree p2)" using poly_add_length_le by (meson diff_le_mono le_max_iff_disj)
lemma poly_add_coeff_aux: assumes"length p1 ≥ length p2" shows"coeff (poly_add p1 p2) = (λi. ((coeff p1) i) ⊕ ((coeff p2) i))" proof fix i have"i < length p1 ==> (coeff (poly_add p1 p2)) i = ((coeff p1) i) ⊕ ((coeff p2) i)" proof - let ?p2 = "(replicate (length p1 - length p2) 0) @ p2" have len_eqs: "length p1 = length ?p2""length (map2 (⊕) p1 ?p2) = length p1" using assms by auto assume i_lt: "i < length p1" have"(coeff (poly_add p1 p2)) i = (coeff (map2 (⊕) p1 ?p2)) i" using normalize_coeff[of "map2 (⊕) p1 ?p2"] assms by auto alsohave" ... = (map2 (⊕) p1 ?p2) ! (length p1 - 1 - i)" using coeff_nth[of i "map2 (⊕) p1 ?p2"] len_eqs(2) i_lt by auto alsohave" ... = (p1 ! (length p1 - 1 - i)) ⊕ (?p2 ! (length ?p2 - 1 - i))" using len_eqs i_lt by auto alsohave" ... = ((coeff p1) i) ⊕ ((coeff ?p2) i)" using coeff_nth[of i p1] coeff_nth[of i ?p2] i_lt len_eqs(1) by auto alsohave" ... = ((coeff p1) i) ⊕ ((coeff p2) i)" using prefix_replicate_zero_coeff by simp finallyshow"(coeff (poly_add p1 p2)) i = ((coeff p1) i) ⊕ ((coeff p2) i)" . qed moreover have"i ≥ length p1 ==> (coeff (poly_add p1 p2)) i = ((coeff p1) i) ⊕ ((coeff p2) i)" using coeff_length[of "poly_add p1 p2"] coeff_length[of p1] coeff_length[of p2]
poly_add_length_le[of p1 p2] assms by auto ultimatelyshow"(coeff (poly_add p1 p2)) i = ((coeff p1) i) ⊕ ((coeff p2) i)" using not_le by blast qed
lemma poly_add_coeff: assumes"set p1 ⊆ carrier R""set p2 ⊆ carrier R" shows"coeff (poly_add p1 p2) = (λi. ((coeff p1) i) ⊕ ((coeff p2) i))" proof - have"length p1 ≥ length p2 ∨ length p2 > length p1" by auto thus ?thesis proof assume"length p1 ≥ length p2"thus ?thesis using poly_add_coeff_aux by simp next assume"length p2 > length p1" hence"coeff (poly_add p1 p2) = (λi. ((coeff p2) i) ⊕ ((coeff p1) i))" using poly_add_coeff_aux by simp thus ?thesis using assms by (simp add: add.m_comm) qed qed
lemma poly_add_comm: assumes"set p1 ⊆ carrier R""set p2 ⊆ carrier R" shows"poly_add p1 p2 = poly_add p2 p1" proof - have"coeff (poly_add p1 p2) = coeff (poly_add p2 p1)" using poly_add_coeff[OF assms] poly_add_coeff[OF assms(2) assms(1)]
coeff_in_carrier[OF assms(1)] coeff_in_carrier[OF assms(2)] add.m_comm by auto thus ?thesis using coeff_iff_polynomial_cond[OF
poly_add_is_polynomial[OF carrier_is_subring assms]
poly_add_is_polynomial[OF carrier_is_subring assms(2,1)]] by simp qed
lemma poly_add_monom: assumes"set p ⊆ carrier R"and"a ∈ carrier R - { 0 }" shows"poly_add (monom a (length p)) p = a # p" unfolding monom_def using assms by (induction p) (auto)
lemma poly_add_append_replicate: assumes"set p ⊆ carrier R""set q ⊆ carrier R" shows"poly_add (p @ (replicate (length q) 0)) q = normalize (p @ q)" proof - have"map2 (⊕) (p @ (replicate (length q) 0)) ((replicate (length p) 0) @ q) = p @ q" using assms by (induct p) (induct q, auto) thus ?thesis by simp qed
lemma poly_add_append_zero: assumes"set p ⊆ carrier R""set q ⊆ carrier R" shows"poly_add (p @ [ 0 ]) (q @ [ 0 ]) = normalize ((poly_add p q) @ [ 0 ])" proof - have in_carrier: "set (p @ [ 0 ]) ⊆ carrier R""set (q @ [ 0 ]) ⊆ carrier R" using assms by auto have"coeff (poly_add (p @ [ 0 ]) (q @ [ 0 ])) = coeff ((poly_add p q) @ [ 0 ])" using append_coeff[of p "[ 0 ]"] poly_add_coeff[OF in_carrier]
append_coeff[of q "[ 0 ]"] append_coeff[of "poly_add p q""[ 0 ]"]
poly_add_coeff[OF assms] assms[THEN coeff_in_carrier] by auto hence"coeff (poly_add (p @ [ 0 ]) (q @ [ 0 ])) = coeff (normalize ((poly_add p q) @ [ 0 ]))" using normalize_coeff by simp moreoverhave"set ((poly_add p q) @ [ 0 ]) ⊆ carrier R" using poly_add_in_carrier[OF assms] by simp ultimatelyshow ?thesis using coeff_iff_polynomial_cond[OF poly_add_is_polynomial[OF carrier_is_subring in_carrier]
normalize_gives_polynomial] by simp qed
lemma poly_add_normalize_aux: assumes"set p1 ⊆ carrier R""set p2 ⊆ carrier R" shows"poly_add p1 p2 = poly_add (normalize p1) p2" proof - have aux_lemma: "poly_add p1 p2 = poly_add ((replicate n 0) @ p1) p2" if"set p1 ⊆ carrier R""set p2 ⊆ carrier R" for n p1 p2 using that proof (induction n) case 0 thus ?caseby simp next case (Suc n) have aux_lemma: "poly_add p1 p2 = poly_add (0 # p1) p2" if in_carrier: "set p1 ⊆ carrier R""set p2 ⊆ carrier R" for p1 p2 proof - have"length p1 ≥ length p2 ==> ?thesis" proof - assume A: "length p1 ≥ length p2" let ?p2 = "λn. (replicate n 0) @ p2" have"poly_add p1 p2 = normalize (map2 (⊕) (0 # p1) (0 # ?p2 (length p1 - length p2)))" using A by simp alsohave" ... = normalize (map2 (⊕) (0 # p1) (?p2 (length (0 # p1) - length p2)))" by (simp add: A Suc_diff_le) alsohave" ... = poly_add (0 # p1) p2" using A by simp finallyshow ?thesis . qed moreoverhave"length p2 > length p1 ==> ?thesis" proof - assume A: "length p2 > length p1" let ?f = "λn p. (replicate n 0) @ p" have"poly_add p1 p2 = poly_add p2 p1" using A by simp alsohave" ... = normalize (map2 (⊕) p2 (?f (length p2 - length p1) p1))" using A by simp alsohave" ... = normalize (map2 (⊕) p2 (?f (length p2 - Suc (length p1)) (0 # p1)))" by (metis A Suc_diff_Suc append_Cons replicate_Suc replicate_app_Cons_same) alsohave" ... = poly_add p2 (0 # p1)" using A by simp alsohave" ... = poly_add (0 # p1) p2" using poly_add_comm[of p2 "0 # p1"] in_carrier by auto finallyshow ?thesis . qed ultimatelyshow ?thesis by auto qed
from Suc have in_carrier: "set (replicate n 0 @ p1) ⊆ carrier R" by auto have"poly_add p1 p2 = poly_add (replicate n 0 @ p1) p2" using Suc by simp alsohave" ... = poly_add (replicate (Suc n) 0 @ p1) p2" using aux_lemma[OF in_carrier Suc(3)] by simp finallyshow ?case . qed
have"poly_add p1 p2 = poly_add ((replicate (length p1 - length (normalize p1)) 0) @ normalize p1) p2" using normalize_def'[of p1] by simp alsohave" ... = poly_add (normalize p1) p2" using aux_lemma[OF normalize_in_carrier[OF assms(1)] assms(2)] by simp finallyshow ?thesis . qed
lemma poly_add_zero': assumes"set p ⊆ carrier R" shows"poly_add p [] = normalize p"and"poly_add [] p = normalize p" proof - have"map2 (⊕) p (replicate (length p) 0) = p" using assms by (induct p) (auto) thus"poly_add p [] = normalize p"and"poly_add [] p = normalize p" using poly_add_comm[OF assms, of "[]"] by simp+ qed
lemma poly_add_zero: assumes"subring K R""polynomial K p" shows"poly_add p [] = p"and"poly_add [] p = p" using poly_add_zero' normalize_polynomial polynomial_in_carrier assms by auto
lemma poly_add_replicate_zero': assumes"set p ⊆ carrier R" shows"poly_add p (replicate n 0) = normalize p"and"poly_add (replicate n 0) p = normalize p" proof - have"poly_add p (replicate n 0) = poly_add p []" using poly_add_normalize(2)[OF assms, of "replicate n 0"]
normalize_replicate_zero[of n "[]"] by force alsohave" ... = normalize p" using poly_add_zero'[OF assms] by simp finallyshow"poly_add p (replicate n 0) = normalize p" . thus"poly_add (replicate n 0) p = normalize p" using poly_add_comm[OF assms, of "replicate n 0"] by force qed
lemma poly_add_replicate_zero: assumes"subring K R""polynomial K p" shows"poly_add p (replicate n 0) = p"and"poly_add (replicate n 0) p = p" using poly_add_replicate_zero' normalize_polynomial polynomial_in_carrier assms by auto
subsection‹Dense Representation›
lemma dense_repr_replicate_zero: "dense_repr ((replicate n 0) @ p) = dense_repr p" by (induction n) (auto)
lemma polynomial_dense_repr: assumes"polynomial K p"and"p ≠ []" shows"dense_repr p = (lead_coeff p, degree p) # dense_repr (normalize (tl p))" proof - let ?len = length and ?norm = normalize obtain a p' where p: "p = a # p'" using assms(2) list.exhaust_sel by blast hence a: "a ∈ K - { 0 }"and p': "set p' ⊆ K" using assms(1) unfolding p by (auto simp add: polynomial_def) hence"dense_repr p = (lead_coeff p, degree p) # dense_repr p'" unfolding p by simp alsohave" ... = (lead_coeff p, degree p) # dense_repr ((replicate (?len p' - ?len (?norm p')) 0) @ ?norm p')" using normalize_def' dense_repr_replicate_zero by simp alsohave" ... = (lead_coeff p, degree p) # dense_repr (?norm p')" using dense_repr_replicate_zero by simp finallyshow ?thesis unfolding p by simp qed
lemma monom_decomp: assumes"subring K R""polynomial K p" shows"p = poly_of_dense (dense_repr p)" using assms(2) proof (induct "length p" arbitrary: p rule: less_induct) case less thus ?case proof (cases p) case Nil thus ?thesis by simp next case (Cons a l) hence a: "a ∈ carrier R - { 0 }"and l: "set l ⊆ carrier R""set l ⊆ K" using less(2) subringE(1)[OF assms(1)] by (auto simp add: polynomial_def) hence"a # l = poly_add (monom a (degree (a # l))) l" using poly_add_monom[of l a] by simp alsohave" ... = poly_add (monom a (degree (a # l))) (normalize l)" using poly_add_normalize(2)[of "monom a (degree (a # l))", OF _ l(1)] a unfolding monom_def by force alsohave" ... = poly_add (monom a (degree (a # l))) (poly_of_dense (dense_repr (normalize l)))" using less(1)[OF _ normalize_gives_polynomial[OF l(2)]] normalize_length_le[of l] unfolding Cons by simp alsohave" ... = poly_of_dense ((a, degree (a # l)) # dense_repr (normalize l))" by simp alsohave" ... = poly_of_dense (dense_repr (a # l))" using polynomial_dense_repr[OF less(2)] unfolding Cons by simp finallyshow ?thesis unfolding Cons by simp qed qed
subsection‹Polynomial Multiplication›
lemma poly_mult_is_polynomial: assumes"subring K R""set p1 ⊆ K"and"set p2 ⊆ K" shows"polynomial K (poly_mult p1 p2)" using assms(2-3) proof (induction p1) case Nil thus ?case by (simp add: polynomial_def) next case (Cons a p1) let ?a_p2 = "(map (λb. a ⊗ b) p2) @ (replicate (degree (a # p1)) 0)"
have"set (poly_mult p1 p2) ⊆ K" using Cons unfolding polynomial_def by auto moreoverhave"set ?a_p2 ⊆ K" using assms(3) Cons(2) subringE(1-2,6)[OF assms(1)] by(induct p2) (auto) ultimatelyhave"polynomial K (poly_add ?a_p2 (poly_mult p1 p2))" using poly_add_is_polynomial[OF assms(1)] by blast thus ?caseby simp qed
lemma poly_mult_closed: assumes"subring K R" shows"[ polynomial K p1; polynomial K p2 ]==> polynomial K (poly_mult p1 p2)" using poly_mult_is_polynomial polynomial_incl assms by simp
lemma poly_mult_in_carrier: "[ set p1 ⊆ carrier R; set p2 ⊆ carrier R ]==> set (poly_mult p1 p2) ⊆ carrier R" using poly_mult_is_polynomial polynomial_in_carrier carrier_is_subring by simp
lemma poly_mult_coeff: assumes"set p1 ⊆ carrier R""set p2 ⊆ carrier R" shows"coeff (poly_mult p1 p2) = (λi. ⨁ k ∈ {..i}. (coeff p1) k ⊗ (coeff p2) (i - k))" using assms(1) proof (induction p1) case Nil thus ?caseusing assms(2) by auto next case (Cons a p1) hence in_carrier: "a ∈ carrier R""∧i. (coeff p1) i ∈ carrier R""∧i. (coeff p2) i ∈ carrier R" using coeff_in_carrier assms(2) by auto
let ?a_p2 = "(map (λb. a ⊗ b) p2) @ (replicate (degree (a # p1)) 0)" have"coeff (replicate (degree (a # p1)) 0) = (λ_. 0)" and"length (replicate (degree (a # p1)) 0) = length p1" using prefix_replicate_zero_coeff[of "[]""length p1"] by auto hence"coeff ?a_p2 = (λi. if i < length p1 then 0 else (coeff (map (λb. a ⊗ b) p2)) (i - length p1))" using append_coeff[of "map (λb. a ⊗ b) p2""replicate (length p1) 0"] by auto alsohave" ... = (λi. if i < length p1 then 0 else a ⊗ ((coeff p2) (i - length p1)))" proof - have"∧i. i < length p2 ==> (coeff (map (λb. a ⊗ b) p2)) i = a ⊗ ((coeff p2) i)" proof - fix i assume i_lt: "i < length p2" hence"(coeff (map (λb. a ⊗ b) p2)) i = (map (λb. a ⊗ b) p2) ! (length p2 - 1 - i)" using coeff_nth[of i "map (λb. a ⊗ b) p2"] by auto alsohave" ... = a ⊗ (p2 ! (length p2 - 1 - i))" using i_lt by auto alsohave" ... = a ⊗ ((coeff p2) i)" using coeff_nth[OF i_lt] by simp finallyshow"(coeff (map (λb. a ⊗ b) p2)) i = a ⊗ ((coeff p2) i)" . qed moreoverhave"∧i. i ≥ length p2 ==> (coeff (map (λb. a ⊗ b) p2)) i = a ⊗ ((coeff p2) i)" using coeff_length[of p2] coeff_length[of "map (λb. a ⊗ b) p2"] in_carrier by auto ultimatelyshow ?thesis by (meson not_le) qed alsohave" ... = (λi. ⨁ k ∈ {..i}. (if k = length p1 then a else 0) ⊗ (coeff p2) (i - k))"
(is"?f1 = (λi. (⨁ k ∈ {..i}. ?f2 k ⊗ ?f3 (i - k)))") proof fix i have"∧k. k ∈ {..i} ==> ?f2 k ⊗ ?f3 (i - k) = 0"if"i < length p1" using in_carrier that by auto hence"(⨁ k ∈ {..i}. ?f2 k ⊗ ?f3 (i - k)) = 0"if"i < length p1" using that in_carrier
add.finprod_cong'[of "{..i}""{..i}""λk. ?f2 k ⊗ ?f3 (i - k)""λi. 0"] by auto hence eq_lt: "?f1 i = (λi. (⨁ k ∈ {..i}. ?f2 k ⊗ ?f3 (i - k))) i"if"i < length p1" using that by auto
have"∧k. k ∈ {..i} ==> ?f2 k ⊗🪙R🪙 ?f3 (i - k) = (if length p1 = k then a ⊗ coeff p2 (i - k) else 0)" using in_carrier by auto hence"(⨁ k ∈ {..i}. ?f2 k ⊗ ?f3 (i - k)) = (⨁ k ∈ {..i}. (if length p1 = k then a ⊗ coeff p2 (i - k) else 0))" using in_carrier
add.finprod_cong'[of "{..i}""{..i}""λk. ?f2 k ⊗ ?f3 (i - k)" "λk. (if length p1 = k then a ⊗ coeff p2 (i - k) else 0)"] by fastforce alsohave" ... = a ⊗ (coeff p2) (i - length p1)"if"i ≥ length p1" using add.finprod_singleton[of "length p1""{..i}""λj. a ⊗ (coeff p2) (i - j)"]
in_carrier that by auto finally have"(⨁ k ∈ {..i}. ?f2 k ⊗ ?f3 (i - k)) = a ⊗ (coeff p2) (i - length p1)"if"i ≥ length p1" using that by simp hence eq_ge: "?f1 i = (λi. (⨁ k ∈ {..i}. ?f2 k ⊗ ?f3 (i - k))) i"if"i ≥ length p1" using that by auto
from eq_lt eq_ge show"?f1 i = (λi. (⨁ k ∈ {..i}. ?f2 k ⊗ ?f3 (i - k))) i"by auto qed
finallyhave coeff_a_p2: "coeff ?a_p2 = (λi. ⨁ k ∈ {..i}. (if k = length p1 then a else 0) ⊗ (coeff p2) (i - k))" .
have"set ?a_p2 ⊆ carrier R" using in_carrier(1) assms(2) by auto
moreoverhave"set (poly_mult p1 p2) ⊆ carrier R" using poly_mult_in_carrier[OF _ assms(2)] Cons(2) by simp
ultimately have"coeff (poly_mult (a # p1) p2) = (λi. ((coeff ?a_p2) i) ⊕ ((coeff (poly_mult p1 p2)) i))" using poly_add_coeff[of ?a_p2 "poly_mult p1 p2"] by simp alsohave" ... = (λi. (⨁ k ∈ {..i}. (if k = length p1 then a else 0) ⊗ (coeff p2) (i - k)) ⊕ (⨁ k ∈ {..i}. (coeff p1) k ⊗ (coeff p2) (i - k)))" using Cons coeff_a_p2 by simp alsohave" ... = (λi. (⨁ k ∈ {..i}. ((if k = length p1 then a else 0) ⊗ (coeff p2) (i - k)) ⊕ ((coeff p1) k ⊗ (coeff p2) (i - k))))" using add.finprod_multf in_carrier by auto alsohave" ... = (λi. (⨁ k ∈ {..i}. (coeff (a # p1) k) ⊗ (coeff p2) (i - k)))"
(is"(λi. (⨁ k ∈ {..i}. ?f i k)) = (λi. (⨁ k ∈ {..i}. ?g i k))") proof fix i have"∧k. ?f i k = ?g i k" using in_carrier coeff_length[of p1] by auto thus"(⨁ k ∈ {..i}. ?f i k) = (⨁ k ∈ {..i}. ?g i k)"by simp qed finallyshow ?case . qed
lemma poly_mult_zero: assumes"set p ⊆ carrier R" shows"poly_mult [] p = []"and"poly_mult p [] = []" proof (simp) have"coeff (poly_mult p []) = (λ_. 0)" using poly_mult_coeff[OF assms, of "[]"] coeff_in_carrier[OF assms] by auto thus"poly_mult p [] = []" using coeff_iff_polynomial_cond[OF
poly_mult_is_polynomial[OF carrier_is_subring assms] zero_is_polynomial] by simp qed
lemma poly_mult_l_distr': assumes"set p1 ⊆ carrier R""set p2 ⊆ carrier R""set p3 ⊆ carrier R" shows"poly_mult (poly_add p1 p2) p3 = poly_add (poly_mult p1 p3) (poly_mult p2 p3)" proof - let ?c1 = "coeff p1"and ?c2 = "coeff p2"and ?c3 = "coeff p3" have in_carrier: "∧i. ?c1 i ∈ carrier R""∧i. ?c2 i ∈ carrier R""∧i. ?c3 i ∈ carrier R" using assms coeff_in_carrier by auto
have"coeff (poly_mult (poly_add p1 p2) p3) = (λn. ⨁i ∈ {..n}. (?c1 i ⊕ ?c2 i) ⊗ ?c3 (n - i))" using poly_mult_coeff[of "poly_add p1 p2" p3] poly_add_coeff[OF assms(1-2)]
poly_add_in_carrier[OF assms(1-2)] assms by auto alsohave" ... = (λn. ⨁i ∈ {..n}. (?c1 i ⊗ ?c3 (n - i)) ⊕ (?c2 i ⊗ ?c3 (n - i)))" using in_carrier l_distr by auto also have" ... = (λn. (⨁i ∈ {..n}. (?c1 i ⊗ ?c3 (n - i))) ⊕ (⨁i ∈ {..n}. (?c2 i ⊗ ?c3 (n - i))))" using add.finprod_multf in_carrier by auto alsohave" ... = coeff (poly_add (poly_mult p1 p3) (poly_mult p2 p3))" using poly_mult_coeff[OF assms(1) assms(3)] poly_mult_coeff[OF assms(2-3)]
poly_add_coeff[OF poly_mult_in_carrier[OF assms(1) assms(3)]]
poly_mult_in_carrier[OF assms(2-3)] by simp finallyhave"coeff (poly_mult (poly_add p1 p2) p3) = coeff (poly_add (poly_mult p1 p3) (poly_mult p2 p3))" . moreoverhave"polynomial (carrier R) (poly_mult (poly_add p1 p2) p3)" and"polynomial (carrier R) (poly_add (poly_mult p1 p3) (poly_mult p2 p3))" using assms poly_add_is_polynomial poly_mult_is_polynomial polynomial_in_carrier
carrier_is_subring by auto ultimatelyshow ?thesis using coeff_iff_polynomial_cond by auto qed
lemma poly_mult_l_distr: assumes"subring K R""polynomial K p1""polynomial K p2""polynomial K p3" shows"poly_mult (poly_add p1 p2) p3 = poly_add (poly_mult p1 p3) (poly_mult p2 p3)" using poly_mult_l_distr' polynomial_in_carrier assms by auto
lemma poly_mult_prepend_replicate_zero: assumes"set p1 ⊆ carrier R""set p2 ⊆ carrier R" shows"poly_mult p1 p2 = poly_mult ((replicate n 0) @ p1) p2" proof - have aux_lemma: "poly_mult p1 p2 = poly_mult (0 # p1) p2" if A: "set p1 ⊆ carrier R""set p2 ⊆ carrier R"for p1 p2 proof - let ?a_p2 = "(map ((⊗) 0) p2) @ (replicate (length p1) 0)" have"?a_p2 = replicate (length p2 + length p1) 0" using A(2) by (induction p2) (auto) hence"poly_mult (0 # p1) p2 = poly_add (replicate (length p2 + length p1) 0) (poly_mult p1 p2)" by simp alsohave" ... = poly_add (normalize (replicate (length p2 + length p1) 0)) (poly_mult p1 p2)" using poly_add_normalize(1)[of "replicate (length p2 + length p1) 0""poly_mult p1 p2"]
poly_mult_in_carrier[OF A] by force alsohave" ... = poly_mult p1 p2" using poly_add_zero(2)[OF _ poly_mult_is_polynomial[OF _ A]] carrier_is_subring
normalize_replicate_zero[of "length p2 + length p1""[]"] by simp finallyshow ?thesis by auto qed from assms show ?thesis proof (induction n) case 0 thus ?caseby simp next case (Suc n) thus ?caseusing aux_lemma[of "replicate n 0 @ p1" p2] by force qed qed
lemma poly_mult_normalize: assumes"set p1 ⊆ carrier R""set p2 ⊆ carrier R" shows"poly_mult p1 p2 = poly_mult (normalize p1) p2" proof - let ?replicate = "replicate (length p1 - length (normalize p1)) 0" have"poly_mult p1 p2 = poly_mult (?replicate @ (normalize p1)) p2" using normalize_def'[of p1] by simp thus ?thesis using poly_mult_prepend_replicate_zero normalize_in_carrier assms by auto qed
lemma poly_mult_append_zero: assumes"set p ⊆ carrier R""set q ⊆ carrier R" shows"poly_mult (p @ [ 0 ]) q = normalize ((poly_mult p q) @ [ 0 ])" using assms(1) proof (induct p) case Nil thus ?case using poly_mult_normalize[OF _ assms(2), of "[] @ [ 0 ]"]
poly_mult_zero(1) poly_mult_zero(1)[of "q @ [ 0 ]"] assms(2) by auto next case (Cons a p) let ?q_a = "λn. (map ((⊗) a) q) @ (replicate n 0)" have set_q_a: "∧n. set (?q_a n) ⊆ carrier R" using Cons(2) assms(2) by (induct q) (auto) have set_poly_mult: "set ((poly_mult p q) @ [ 0 ]) ⊆ carrier R" using poly_mult_in_carrier[OF _ assms(2)] Cons(2) by auto have"poly_mult ((a # p) @ [0]) q = poly_add (?q_a (Suc (length p))) (poly_mult (p @ [0]) q)" by auto alsohave" ... = poly_add (?q_a (Suc (length p))) (normalize ((poly_mult p q) @ [0 ]))" using Cons by simp alsohave" ... = poly_add ((?q_a (length p)) @ [ 0 ]) ((poly_mult p q) @ [ 0 ])" using poly_add_normalize(2)[OF set_q_a[of "Suc (length p)"] set_poly_mult] by (simp add: replicate_append_same) alsohave" ... = normalize ((poly_add (?q_a (length p)) (poly_mult p q)) @ [ 0 ])" using poly_add_append_zero[OF set_q_a[of "length p"] poly_mult_in_carrier[OF _ assms(2)]] Cons(2) by auto alsohave" ... = normalize ((poly_mult (a # p) q) @ [ 0 ])" by auto finallyshow ?case . qed
end(* of ring context. *)
subsection‹Properties Within a Domain›
contextdomain begin
lemma one_is_polynomial [intro]: "subring K R ==> polynomial K [ 1 ]" unfolding polynomial_def using subringE(3) by auto
lemma poly_mult_comm: assumes"set p1 ⊆ carrier R""set p2 ⊆ carrier R" shows"poly_mult p1 p2 = poly_mult p2 p1" proof - let ?c1 = "coeff p1"and ?c2 = "coeff p2" have"∧i. (⨁k ∈ {..i}. ?c1 k ⊗ ?c2 (i - k)) = (⨁k ∈ {..i}. ?c2 k ⊗ ?c1 (i - k))" proof - fix i :: nat let ?f = "λk. ?c1 k ⊗ ?c2 (i - k)" have in_carrier: "∧i. ?c1 i ∈ carrier R""∧i. ?c2 i ∈ carrier R" using coeff_in_carrier[OF assms(1)] coeff_in_carrier[OF assms(2)] by auto
have reindex_inj: "inj_on (λk. i - k) {..i}" using inj_on_def by force moreoverhave"(λk. i - k) ` {..i} ⊆ {..i}"by auto hence"(λk. i - k) ` {..i} = {..i}" using reindex_inj endo_inj_surj[of "{..i}""λk. i - k"] by simp ultimatelyhave"(⨁k ∈ {..i}. ?f k) = (⨁k ∈ {..i}. ?f (i - k))" using add.finprod_reindex[of ?f "λk. i - k""{..i}"] in_carrier by auto
moreoverhave"∧k. k ∈ {..i} ==> ?f (i - k) = ?c2 k ⊗ ?c1 (i - k)" using in_carrier m_comm by auto hence"(⨁k ∈ {..i}. ?f (i - k)) = (⨁k ∈ {..i}. ?c2 k ⊗ ?c1 (i - k))" using add.finprod_cong'[of "{..i}""{..i}"] in_carrier by auto ultimatelyshow"(⨁k ∈ {..i}. ?f k) = (⨁k ∈ {..i}. ?c2 k ⊗ ?c1 (i - k))" by simp qed hence"coeff (poly_mult p1 p2) = coeff (poly_mult p2 p1)" using poly_mult_coeff[OF assms] poly_mult_coeff[OF assms(2,1)] by simp thus ?thesis using coeff_iff_polynomial_cond[OF poly_mult_is_polynomial[OF _ assms]
poly_mult_is_polynomial[OF _ assms(2,1)]]
carrier_is_subring by simp qed
lemma poly_mult_r_distr: assumes"subring K R""polynomial K p1""polynomial K p2""polynomial K p3" shows"poly_mult p1 (poly_add p2 p3) = poly_add (poly_mult p1 p2) (poly_mult p1 p3)" using poly_mult_r_distr' polynomial_in_carrier assms by auto
lemma poly_mult_replicate_zero: assumes"set p ⊆ carrier R" shows"poly_mult (replicate n 0) p = []" and"poly_mult p (replicate n 0) = []" proof - have in_carrier: "∧n. set (replicate n 0) ⊆ carrier R"by auto show"poly_mult (replicate n 0) p = []"using assms proof (induction n) case 0 thus ?caseby simp next case (Suc n) hence"poly_mult (replicate (Suc n) 0) p = poly_mult (0 # (replicate n 0)) p" by simp alsohave" ... = poly_add ((map (λa. 0⊗ a) p) @ (replicate n 0)) []" using Suc by simp alsohave" ... = poly_add ((map (λa. 0) p) @ (replicate n 0)) []" proof - have"map ((⊗) 0) p = map (λa. 0) p" using Suc.prems by auto thenshow ?thesis by presburger qed alsohave" ... = poly_add (replicate (length p + n) 0) []" by (simp add: map_replicate_const replicate_add) alsohave" ... = poly_add [] []" using poly_add_normalize(1)[of "replicate (length p + n) 0""[]"]
normalize_replicate_zero[of "length p + n""[]"] by auto alsohave" ... = []"by simp finallyshow ?case . qed thus"poly_mult p (replicate n 0) = []" using poly_mult_comm[OF assms in_carrier] by simp qed
lemma poly_mult_const': assumes"set p ⊆ carrier R""a ∈ carrier R" shows"poly_mult [ a ] p = normalize (map (λb. a ⊗ b) p)" and"poly_mult p [ a ] = normalize (map (λb. a ⊗ b) p)" proof - have"map2 (⊕) (map ((⊗) a) p) (replicate (length p) 0) = map ((⊗) a) p" using assms by (induction p) (auto) thus"poly_mult [ a ] p = normalize (map (λb. a ⊗ b) p)"by simp thus"poly_mult p [ a ] = normalize (map (λb. a ⊗ b) p)" using poly_mult_comm[OF assms(1), of "[ a ]"] assms(2) by auto qed
lemma poly_mult_const: assumes"subring K R""polynomial K p""a ∈ K - { 0 }" shows"poly_mult [ a ] p = map (λb. a ⊗ b) p" and"poly_mult p [ a ] = map (λb. a ⊗ b) p" proof - have in_carrier: "set p ⊆ carrier R""a ∈ carrier R" using polynomial_in_carrier[OF assms(1-2)] assms(3) subringE(1)[OF assms(1)] by auto
show"poly_mult [ a ] p = map (λb. a ⊗ b) p" proof (cases p) case Nil thus ?thesis using poly_mult_const'(1) in_carrier by auto next case (Cons b q) have"lead_coeff (map (λb. a ⊗ b) p) ≠0" using assms subringE(1)[OF assms(1)] integral[of a b] Cons lead_coeff_in_carrier by auto hence"normalize (map (λb. a ⊗ b) p) = (map (λb. a ⊗ b) p)" unfolding Cons by simp thus ?thesis using poly_mult_const'(1) in_carrier by auto qed thus"poly_mult p [ a ] = map (λb. a ⊗ b) p" using poly_mult_comm[OF in_carrier(1)] in_carrier(2) by auto qed
lemma poly_mult_semiassoc: assumes"set p ⊆ carrier R""set q ⊆ carrier R"and"a ∈ carrier R" shows"poly_mult (poly_mult [ a ] p) q = poly_mult [ a ] (poly_mult p q)" proof - let ?cp = "coeff p"and ?cq = "coeff q" have"coeff (poly_mult [ a ] p) = (λi. (a ⊗ ?cp i))" using poly_mult_const'(1)[OF assms(1,3)] normalize_coeff scalar_coeff[OF assms(3)] by simp
hence"coeff (poly_mult (poly_mult [ a ] p) q) = (λi. (⨁j ∈ {..i}. (a ⊗ ?cp j) ⊗ ?cq (i - j)))" using poly_mult_coeff[OF poly_mult_in_carrier[OF _ assms(1)] assms(2), of "[ a ]"] assms(3) by auto alsohave" ... = (λi. a ⊗ (⨁j ∈ {..i}. ?cp j ⊗ ?cq (i - j)))" proof fix i show"(⨁j ∈ {..i}. (a ⊗ ?cp j) ⊗ ?cq (i - j)) = a ⊗ (⨁j ∈ {..i}. ?cp j ⊗ ?cq (i - j))" using finsum_rdistr[OF _ assms(3), of _ "λj. ?cp j ⊗ ?cq (i - j)"]
assms(1-2)[THEN coeff_in_carrier] by (simp add: assms(3) m_assoc) qed alsohave" ... = coeff (poly_mult [ a ] (poly_mult p q))" unfolding poly_mult_const'(1)[OF poly_mult_in_carrier[OF assms(1-2)] assms(3)] using scalar_coeff[OF assms(3), of "poly_mult p q"]
poly_mult_coeff[OF assms(1-2)] normalize_coeff by simp finallyhave"coeff (poly_mult (poly_mult [ a ] p) q) = coeff (poly_mult [ a ] (poly_mult p q))" . moreoverhave"polynomial (carrier R) (poly_mult (poly_mult [ a ] p) q)" and"polynomial (carrier R) (poly_mult [ a ] (poly_mult p q))" using poly_mult_is_polynomial[OF _ poly_mult_in_carrier[OF _ assms(1)] assms(2)]
poly_mult_is_polynomial[OF _ _ poly_mult_in_carrier[OF assms(1-2)]]
carrier_is_subring assms(3) by (auto simp del: poly_mult.simps) ultimatelyshow ?thesis using coeff_iff_polynomial_cond by simp qed
text‹Note that "polynomial (carrier R) p" and "subring K p; polynomial K p" are "equivalent" assumptions for any lemma in ring which the result doesn't depend on K, because carrier is a subring and a polynomial for a subset of the carrier is a carrier polynomial. The decision between one of them should be based on how the lemma is going to be used and proved. These are some tips: (a) Lemmas about the algebraic structure of polynomials should use the latter option. (b) Also, if the lemma deals with lots of polynomials, then the latter option is preferred. (c) If the proof is going to be much easier with the first option, do not hesitate. ›
lemma poly_mult_monom': assumes"set p ⊆ carrier R""a ∈ carrier R" shows"poly_mult (monom a n) p = normalize ((map ((⊗) a) p) @ (replicate n 0))" proof - have set_map: "set ((map ((⊗) a) p) @ (replicate n 0)) ⊆ carrier R" using assms by (induct p) (auto) show ?thesis using poly_mult_replicate_zero(1)[OF assms(1), of n]
poly_add_zero'(1)[OF set_map] unfolding monom_def by simp qed
lemma poly_mult_monom: assumes"polynomial (carrier R) p""a ∈ carrier R - { 0 }" shows"poly_mult (monom a n) p = (if p = [] then [] else (poly_mult [ a ] p) @ (replicate n 0))" proof (cases p) case Nil thus ?thesis using poly_mult_zero(2)[of "monom a n"] assms(2) monom_def by fastforce next case (Cons b ps) hence"lead_coeff ((map (λb. a ⊗ b) p) @ (replicate n 0)) ≠0" using Cons assms integral[of a b] unfolding polynomial_def by auto thus ?thesis using poly_mult_monom'[OF polynomial_incl[OF assms(1)], of a n] assms(2) Cons unfolding poly_mult_const(1)[OF carrier_is_subring assms] by simp qed
lemma poly_mult_one': assumes"set p ⊆ carrier R" shows"poly_mult [ 1 ] p = normalize p"and"poly_mult p [ 1 ] = normalize p" proof - have"map2 (⊕) (map ((⊗) 1) p) (replicate (length p) 0) = p" using assms by (induct p) (auto) thus"poly_mult [ 1 ] p = normalize p"and"poly_mult p [ 1 ] = normalize p" using poly_mult_comm[OF assms, of "[ 1 ]"] by auto qed
lemma poly_mult_one: assumes"subring K R""polynomial K p" shows"poly_mult [ 1 ] p = p"and"poly_mult p [ 1 ] = p" using poly_mult_one'[OF polynomial_in_carrier[OF assms]] normalize_polynomial[OF assms(2)] by auto
lemma poly_mult_lead_coeff_aux: assumes"subring K R""polynomial K p1""polynomial K p2"and"p1 ≠ []"and"p2 ≠ []" shows"(coeff (poly_mult p1 p2)) (degree p1 + degree p2) = (lead_coeff p1) ⊗ (lead_coeff p2)" proof - have p1: "lead_coeff p1 ∈ carrier R - { 0 }"and p2: "lead_coeff p2 ∈ carrier R - { 0 }" using assms(2-5) lead_coeff_in_carrier[OF assms(1)] by (metis list.collapse)+
have"(coeff (poly_mult p1 p2)) (degree p1 + degree p2) = (⨁ k ∈ {..((degree p1) + (degree p2))}. (coeff p1) k ⊗ (coeff p2) ((degree p1) + (degree p2) - k))" using poly_mult_coeff[OF assms(2-3)[THEN polynomial_in_carrier[OF assms(1)]]] by simp alsohave" ... = (lead_coeff p1) ⊗ (lead_coeff p2)" proof - let ?f = "λi. (coeff p1) i ⊗ (coeff p2) ((degree p1) + (degree p2) - i)" have in_carrier: "∧i. (coeff p1) i ∈ carrier R""∧i. (coeff p2) i ∈ carrier R" using coeff_in_carrier assms by auto have"∧i. i < degree p1 ==> ?f i = 0" using coeff_degree[of p2] in_carrier by auto moreoverhave"∧i. i > degree p1 ==> ?f i = 0" using coeff_degree[of p1] in_carrier by auto moreoverhave"?f (degree p1) = (lead_coeff p1) ⊗ (lead_coeff p2)" using assms(4-5) lead_coeff_simp by simp ultimatelyhave"?f = (λi. if degree p1 = i then (lead_coeff p1) ⊗ (lead_coeff p2) else 0)" using nat_neq_iff by auto thus ?thesis using add.finprod_singleton[of "degree p1""{..((degree p1) + (degree p2))}" "λi. (lead_coeff p1) ⊗ (lead_coeff p2)"] p1 p2 by auto qed finallyshow ?thesis . qed
lemma poly_mult_degree_eq: assumes"subring K R""polynomial K p1""polynomial K p2" shows"degree (poly_mult p1 p2) = (if p1 = [] ∨ p2 = [] then 0 else (degree p1) + (degree p2))" proof (cases p1) case Nil thus ?thesis by simp next case (Cons a p1') note p1 = Cons show ?thesis proof (cases p2) case Nil thus ?thesis using poly_mult_zero(2)[OF polynomial_in_carrier[OF assms(1-2)]] by simp next case (Cons b p2') note p2 = Cons have a: "a ∈ carrier R"and b: "b ∈ carrier R" using p1 p2 polynomial_in_carrier[OF assms(1-2)] polynomial_in_carrier[OF assms(1,3)] by auto have"(coeff (poly_mult p1 p2)) ((degree p1) + (degree p2)) = a ⊗ b" using poly_mult_lead_coeff_aux[OF assms] p1 p2 by simp hence neq0: "(coeff (poly_mult p1 p2)) ((degree p1) + (degree p2)) ≠0" using assms(2-3) integral[of a b] lead_coeff_in_carrier[OF assms(1)] p1 p2 by auto moreoverhave eq0: "∧i. i > (degree p1) + (degree p2) ==> (coeff (poly_mult p1 p2)) i = 0" proof - have aux_lemma: "degree (poly_mult p1 p2) ≤ (degree p1) + (degree p2)" proof (induct p1) case Nil thenshow ?caseby simp next case (Cons a p1) let ?a_p2 = "(map (λb. a ⊗ b) p2) @ (replicate (degree (a # p1)) 0)" have"poly_mult (a # p1) p2 = poly_add ?a_p2 (poly_mult p1 p2)"by simp hence"degree (poly_mult (a # p1) p2) ≤ max (degree ?a_p2) (degree (poly_mult p1 p2))" using poly_add_degree[of ?a_p2 "poly_mult p1 p2"] by simp alsohave" ... ≤ max ((degree (a # p1)) + (degree p2)) (degree (poly_mult p1 p2))" by auto alsohave" ... ≤ max ((degree (a # p1)) + (degree p2)) ((degree p1) + (degree p2))" using Cons by simp alsohave" ... ≤ (degree (a # p1)) + (degree p2)" by auto finallyshow ?case . qed fix i show"i > (degree p1) + (degree p2) ==> (coeff (poly_mult p1 p2)) i = 0" using coeff_degree aux_lemma by simp qed moreoverhave"polynomial K (poly_mult p1 p2)" by (simp add: assms poly_mult_closed) ultimatelyhave"degree (poly_mult p1 p2) = degree p1 + degree p2" by (metis (no_types) assms(1) coeff.simps(1) coeff_degree domain.poly_mult_one(1) domain_axioms eq0 lead_coeff_simp length_greater_0_conv neq0 normalize_length_lt not_less_iff_gr_or_eq poly_mult_one'(1) polynomial_in_carrier) thus ?thesis using p1 p2 by auto qed qed
lemma poly_mult_integral: assumes"subring K R""polynomial K p1""polynomial K p2" shows"poly_mult p1 p2 = [] ==> p1 = [] ∨ p2 = []" proof (rule ccontr) assume A: "poly_mult p1 p2 = []""¬ (p1 = [] ∨ p2 = [])" hence"degree (poly_mult p1 p2) = degree p1 + degree p2" using poly_mult_degree_eq[OF assms] by simp hence"length p1 = 1 ∧ length p2 = 1" using A Suc_diff_Suc by fastforce thenobtain a b where p1: "p1 = [ a ]"and p2: "p2 = [ b ]" by (metis One_nat_def length_0_conv length_Suc_conv) hence"a ∈ carrier R - { 0 }"and"b ∈ carrier R - { 0 }" using assms lead_coeff_in_carrier by auto hence"poly_mult [ a ] [ b ] = [ a ⊗ b ]" using integral by auto thus False using A(1) p1 p2 by simp qed
lemma poly_mult_lead_coeff: assumes"subring K R""polynomial K p1""polynomial K p2"and"p1 ≠ []"and"p2 ≠ []" shows"lead_coeff (poly_mult p1 p2) = (lead_coeff p1) ⊗ (lead_coeff p2)" proof - have"poly_mult p1 p2 ≠ []" using poly_mult_integral[OF assms(1-3)] assms(4-5) by auto hence"lead_coeff (poly_mult p1 p2) = (coeff (poly_mult p1 p2)) (degree p1 + degree p2)" using poly_mult_degree_eq[OF assms(1-3)] assms(4-5) by (metis coeff.simps(2) list.collapse) thus ?thesis using poly_mult_lead_coeff_aux[OF assms] by simp qed
lemma poly_mult_append_zero_lcancel: assumes"subring K R"and"polynomial K p""polynomial K q" shows"poly_mult (p @ [ 0 ]) q = r @ [ 0 ] ==> poly_mult p q = r" proof - note in_carrier = assms(2-3)[THEN polynomial_in_carrier[OF assms(1)]]
assume pmult: "poly_mult (p @ [ 0 ]) q = r @ [ 0 ]" have"poly_mult (p @ [ 0 ]) q = []"if"q = []" using poly_mult_zero(2)[of "p @ [ 0 ]"] that in_carrier(1) by auto moreoverhave"poly_mult (p @ [ 0 ]) q = []"if"p = []" using poly_mult_normalize[OF _ in_carrier(2), of "p @ [ 0 ]"] poly_mult_zero[OF in_carrier(2)] unfolding that by auto ultimatelyhave"p ≠ []"and"q ≠ []" using pmult by auto hence"poly_mult p q ≠ []" using poly_mult_integral[OF assms] by auto hence"normalize ((poly_mult p q) @ [ 0 ]) = (poly_mult p q) @ [ 0 ]" using normalize_polynomial[OF append_is_polynomial[OF assms(1) poly_mult_closed[OF assms], of "Suc 0"]] by auto thus"poly_mult p q = r" using poly_mult_append_zero[OF assms(2-3)[THEN polynomial_in_carrier[OF assms(1)]]] pmult by simp qed
lemma poly_mult_append_zero_rcancel: assumes"subring K R"and"polynomial K p""polynomial K q" shows"poly_mult p (q @ [ 0 ]) = r @ [ 0 ] ==> poly_mult p q = r" using poly_mult_append_zero_lcancel[OF assms(1,3,2)]
poly_mult_comm[of p "q @ [ 0 ]"] poly_mult_comm[of p q]
assms(2-3)[THEN polynomial_in_carrier[OF assms(1)]] by auto
end(* of domain context. *)
subsection‹Algebraic Structure of Polynomials›
definition univ_poly :: "('a, 'b) ring_scheme ==>'a set ==> ('a list) ring"
(‹(‹open_block notation=‹postfix X›\🍋)› 80) where"univ_poly R K = ( carrier = { p. polynomial🪙R🪙 K p }, mult = ring.poly_mult R, one = [ 1🪙R🪙 ], zero = [], add = ring.poly_add R )"
text‹These lemmas allow you to unfold one field of the record at a time. ›
lemma univ_poly_carrier: "polynomial🪙R🪙 K p ⟷ p ∈ carrier (K[X]🪙R🪙)" unfolding univ_poly_def by simp
assume A: "polynomial K p1""polynomial K p2""polynomial K p3" show ?P using polynomial_in_carrier[OF K A(1)] proof (induction p1) case Nil thus ?caseby simp next next case (Cons a p1) thus ?case proof (cases "a = 0") assume eq_zero: "a = 0" have p1: "set p1 ⊆ carrier R" using Cons(2) by simp have"poly_mult (poly_mult (a # p1) p2) p3 = poly_mult (poly_mult p1 p2) p3" using poly_mult_prepend_replicate_zero[OF p1 polynomial_in_carrier[OF K A(2)], of "Suc 0"]
eq_zero by simp alsohave" ... = poly_mult p1 (poly_mult p2 p3)" using p1[THEN Cons(1)] by simp alsohave" ... = poly_mult (a # p1) (poly_mult p2 p3)" using poly_mult_prepend_replicate_zero[OF p1
poly_mult_in_carrier[OF A(2-3)[THEN polynomial_in_carrier[OF K]]], of "Suc 0"] eq_zero by simp finallyshow ?thesis . next assume"a ≠0"hence in_carrier: "set p1 ⊆ carrier R""set p2 ⊆ carrier R""set p3 ⊆ carrier R""a ∈ carrier R - { 0 }" using A(2-3) polynomial_in_carrier[OF K] Cons by auto
let ?a_p2 = "(map (λb. a ⊗ b) p2) @ (replicate (length p1) 0)" have a_p2_in_carrier: "set ?a_p2 ⊆ carrier R" using in_carrier by auto
have"poly_mult (poly_mult (a # p1) p2) p3 = poly_mult (poly_add ?a_p2 (poly_mult p1 p2)) p3" by simp alsohave" ... = poly_add (poly_mult ?a_p2 p3) (poly_mult (poly_mult p1 p2) p3)" using poly_mult_l_distr'[OF a_p2_in_carrier poly_mult_in_carrier[OF in_carrier(1-2)] in_carrier(3)] . alsohave" ... = poly_add (poly_mult ?a_p2 p3) (poly_mult p1 (poly_mult p2 p3))" using Cons(1)[OF in_carrier(1)] by simp alsohave" ... = poly_add (poly_mult (normalize ?a_p2) p3) (poly_mult p1 (poly_mult p2 p3))" using poly_mult_normalize[OF a_p2_in_carrier in_carrier(3)] by simp alsohave" ... = poly_add (poly_mult (poly_mult (monom a (length p1)) p2) p3) (poly_mult p1 (poly_mult p2 p3))" using poly_mult_monom'[OF in_carrier(2), of a "length p1"] in_carrier(4) by simp alsohave" ... = poly_add (poly_mult (a # (replicate (length p1) 0)) (poly_mult p2 p3)) (poly_mult p1 (poly_mult p2 p3))" using poly_mult_monom_assoc[of p2 p3 a "length p1"] in_carrier unfolding monom_def by simp alsohave" ... = poly_mult (poly_add (a # (replicate (length p1) 0)) p1) (poly_mult p2 p3)" using poly_mult_l_distr'[of "a # (replicate (length p1) 0)" p1 "poly_mult p2 p3"]
poly_mult_in_carrier[OF in_carrier(2-3)] in_carrier by force alsohave" ... = poly_mult (a # p1) (poly_mult p2 p3)" using poly_add_monom[OF in_carrier(1) in_carrier(4)] unfolding monom_def by simp finallyshow ?thesis . qed qed qed
declare poly_add.simps[simp del]
lemma univ_poly_is_abelian_monoid: "abelian_monoid (K[X])" unfolding univ_poly_def using poly_add_closed poly_add_zero zero_is_polynomial K proof (auto simp add: abelian_monoid_def comm_monoid_def monoid_def comm_monoid_axioms_def) fix p1 p2 p3 let ?c = "λp. coeff p" assume A: "polynomial K p1""polynomial K p2""polynomial K p3" hence
p1: "∧i. (?c p1) i ∈ carrier R""set p1 ⊆ carrier R"and
p2: "∧i. (?c p2) i ∈ carrier R""set p2 ⊆ carrier R"and
p3: "∧i. (?c p3) i ∈ carrier R""set p3 ⊆ carrier R" using A[THEN polynomial_in_carrier[OF K]] coeff_in_carrier by auto have"?c (poly_add (poly_add p1 p2) p3) = (λi. (?c p1 i ⊕ ?c p2 i) ⊕ (?c p3 i))" using poly_add_coeff[OF poly_add_in_carrier[OF p1(2) p2(2)] p3(2)]
poly_add_coeff[OF p1(2) p2(2)] by simp alsohave" ... = (λi. (?c p1 i) ⊕ ((?c p2 i) ⊕ (?c p3 i)))" using p1 p2 p3 add.m_assoc by simp alsohave" ... = ?c (poly_add p1 (poly_add p2 p3))" using poly_add_coeff[OF p1(2) poly_add_in_carrier[OF p2(2) p3(2)]]
poly_add_coeff[OF p2(2) p3(2)] by simp finallyhave"?c (poly_add (poly_add p1 p2) p3) = ?c (poly_add p1 (poly_add p2 p3))" . thus"poly_add (poly_add p1 p2) p3 = poly_add p1 (poly_add p2 p3)" using coeff_iff_polynomial_cond poly_add_closed[OF K] A by meson show"poly_add p1 p2 = poly_add p2 p1" using poly_add_comm[OF p1(2) p2(2)] . qed
lemma univ_poly_is_abelian_group: "abelian_group (K[X])" proof - interpret abelian_monoid "K[X]" using univ_poly_is_abelian_monoid . show ?thesis proof (unfold_locales) show"carrier (add_monoid (K[X])) ⊆ Units (add_monoid (K[X]))" unfolding univ_poly_def Units_def proof (auto) fix p assume p: "polynomial K p" have"polynomial K [ ⊖1 ]" unfolding polynomial_def using r_neg subringE(3,5)[OF K] by force hence cond0: "polynomial K (poly_mult [ ⊖1 ] p)" using poly_mult_closed[OF K, of "[ ⊖1 ]" p] p by simp
have"poly_add p (poly_mult [ ⊖1 ] p) = poly_add (poly_mult [ 1 ] p) (poly_mult [ ⊖1 ] p)" using poly_mult_one[OF K p] by simp alsohave" ... = poly_mult (poly_add [ 1 ] [ ⊖1 ]) p" using poly_mult_l_distr' polynomial_in_carrier[OF K p] by auto alsohave" ... = poly_mult [] p" using poly_add.simps[of "[ 1 ]""[ ⊖1 ]"] by (simp add: case_prod_unfold r_neg) alsohave" ... = []"by simp finallyhave cond1: "poly_add p (poly_mult [ ⊖1 ] p) = []" .
have"poly_add (poly_mult [ ⊖1 ] p) p = poly_add (poly_mult [ ⊖1 ] p) (poly_mult [ 1 ] p)" using poly_mult_one[OF K p] by simp alsohave" ... = poly_mult (poly_add [ ⊖1 ] [ 1 ]) p" using poly_mult_l_distr' polynomial_in_carrier[OF K p] by auto alsohave" ... = poly_mult [] p" using‹poly_mult (poly_add [1] [⊖1]) p = poly_mult [] p› poly_add_comm by auto alsohave" ... = []"by simp finallyhave cond2: "poly_add (poly_mult [ ⊖1 ] p) p = []" .
from cond0 cond1 cond2 show"∃q. polynomial K q ∧ poly_add q p = [] ∧ poly_add p q = []" by auto qed qed qed
lemma univ_poly_is_ring: "ring (K[X])" proof - interpret UP: abelian_group "K[X]" + monoid "K[X]" using univ_poly_is_abelian_group univ_poly_is_monoid . show ?thesis by (unfold_locales)
(auto simp add: univ_poly_def poly_mult_r_distr[OF K] poly_mult_l_distr[OF K]) qed
lemma univ_poly_is_cring: "cring (K[X])" proof - interpret UP: ring "K[X]" using univ_poly_is_ring . have"∧p q. [ p ∈ carrier (K[X]); q ∈ carrier (K[X]) ]==> p ⊗🪙K[X]🪙 q = q ⊗🪙K[X]🪙 p" unfolding univ_poly_def using poly_mult_comm polynomial_in_carrier[OF K] by auto thus ?thesis by unfold_locales auto qed
lemma univ_poly_is_domain: "domain (K[X])" proof - interpret UP: cring "K[X]" using univ_poly_is_cring . show ?thesis by (unfold_locales, auto simp add: univ_poly_def poly_mult_integral[OF K]) qed
declare poly_add.simps[simp]
lemma univ_poly_a_inv_def': assumes"p ∈ carrier (K[X])"shows"⊖🪙K[X]🪙 p = map (λa. ⊖ a) p" proof - have aux_lemma: "∧p. p ∈ carrier (K[X]) ==> p ⊕🪙K[X]🪙 (map (λa. ⊖ a) p) = []" "∧p. p ∈ carrier (K[X]) ==> (map (λa. ⊖ a) p) ∈ carrier (K[X])" proof - fix p assume p: "p ∈ carrier (K[X])" hence set_p: "set p ⊆ K" unfolding univ_poly_def using polynomial_incl by auto show"(map (λa. ⊖ a) p) ∈ carrier (K[X])" proof (cases "p = []") assume"p = []"thus ?thesis unfolding univ_poly_def polynomial_def by auto next assume not_nil: "p ≠ []" hence"lead_coeff p ≠0" using p unfolding univ_poly_def polynomial_def by auto moreoverhave"lead_coeff (map (λa. ⊖ a) p) = ⊖ (lead_coeff p)" using not_nil by (simp add: hd_map) ultimatelyhave"lead_coeff (map (λa. ⊖ a) p) ≠0" using hd_in_set local.minus_zero not_nil set_p subringE(1)[OF K] by force moreoverhave"set (map (λa. ⊖ a) p) ⊆ K" using set_p subringE(5)[OF K] by (induct p) (auto) ultimatelyshow ?thesis unfolding univ_poly_def polynomial_def by simp qed
have"map2 (⊕) p (map (λa. ⊖ a) p) = replicate (length p) 0" using set_p subringE(1)[OF K] by (induct p) (auto simp add: r_neg) thus"p ⊕🪙K[X]🪙 (map (λa. ⊖ a) p) = []" unfolding univ_poly_def using normalize_replicate_zero[of "length p""[]"] by auto qed
interpret UP: ring "K[X]" using univ_poly_is_ring .
from aux_lemma have"∧p. p ∈ carrier (K[X]) ==>⊖🪙K[X]🪙 p = map (λa. ⊖ a) p" by (metis Nil_is_map_conv UP.add.inv_closed UP.l_zero UP.r_neg1 UP.r_zero UP.zero_closed) thus ?thesis using assms by simp qed
(* NEW ========== *) corollary univ_poly_a_inv_length: assumes"p ∈ carrier (K[X])"shows"length (⊖🪙K[X]🪙 p) = length p" unfolding univ_poly_a_inv_def'[OF assms] by simp
(* NEW ========== *) corollary univ_poly_a_inv_degree: assumes"p ∈ carrier (K[X])"shows"degree (⊖🪙K[X]🪙 p) = degree p" using univ_poly_a_inv_length[OF assms] by simp
subsection‹Long Division Theorem›
lemma long_division_theorem: assumes"polynomial K p"and"polynomial K b""b ≠ []" and"lead_coeff b ∈ Units (R ( carrier := K ))" shows"∃q r. polynomial K q ∧ polynomial K r ∧ p = (b ⊗🪙K[X]🪙 q) ⊕🪙K[X]🪙 r ∧ (r = [] ∨ degree r < degree b)"
(is"∃q r. ?long_division p q r") using assms(1) proof (induct "length p" arbitrary: p rule: less_induct) case less thus ?case proof (cases p) case Nil hence"?long_division p [] []" using zero_is_polynomial poly_mult_zero[OF polynomial_in_carrier[OF K assms(2)]] by (simp add: univ_poly_def) thus ?thesis by blast next case (Cons a p') thus ?thesis proof (cases "length b > length p") assume"length b > length p" hence"p = [] ∨ degree p < degree b" by (meson diff_less_mono length_0_conv less_one not_le) hence"?long_division p [] p" using poly_mult_zero(2)[OF polynomial_in_carrier[OF K assms(2)]]
poly_add_zero(2)[OF K less(2)] zero_is_polynomial less(2) by (simp add: univ_poly_def) thus ?thesis by blast next interpret UP: cring "K[X]" using univ_poly_is_cring .
assume"¬ length b > length p" hence len_ge: "length p ≥ length b"by simp obtain c b' where b: "b = c # b'" using assms(3) list.exhaust_sel by blast thenobtain c' where c': "c' ∈ carrier R""c' ∈ K""c' ⊗ c = 1""c ⊗ c' = 1" using assms(4) subringE(1)[OF K] unfolding Units_def by auto have c: "c ∈ carrier R""c ∈ K""c ≠0"and a: "a ∈ carrier R""a ∈ K""a ≠0" using less(2) assms(2) lead_coeff_not_zero subringE(1)[OF K] b Cons by auto hence lc: "c' ⊗ (⊖ a) ∈ K - { 0 }" using subringE(5-6)[OF K] c' add.inv_solve_right integral_iff by fastforce
let ?len = "length"
define s where"s = monom (c' ⊗ (⊖ a)) (?len p - ?len b)" hence s: "polynomial K s""s ≠ []""degree s = ?len p - ?len b""length s ≥ 1" using monom_is_polynomial[OF K lc] unfolding monom_def by auto hence is_polynomial: "polynomial K (p ⊕🪙K[X]🪙 (b ⊗🪙K[X]🪙 s))" using poly_add_closed[OF K less(2) poly_mult_closed[OF K assms(2), of s]] by (simp add: univ_poly_def)
have"lead_coeff (b ⊗🪙K[X]🪙 s) = ⊖ a" using poly_mult_lead_coeff[OF K assms(2) s(1) assms(3) s(2)] c c' a unfolding b s_def monom_def univ_poly_def by (auto simp del: poly_mult.simps, algebra) thenobtain s' where s': "b ⊗🪙K[X]🪙 s = (⊖ a) # s'" using poly_mult_integral[OF K assms(2) s(1)] assms(2-3) s(2) by (simp add: univ_poly_def, metis hd_Cons_tl) moreoverhave"degree p = degree (b ⊗🪙K[X]🪙 s)" using poly_mult_degree_eq[OF K assms(2) s(1)] assms(3) s(2-4) len_ge b Cons by (auto simp add: univ_poly_def) hence"?len p = ?len (b ⊗🪙K[X]🪙 s)" unfolding Cons s' by simp hence"?len (p ⊕🪙K[X]🪙 (b ⊗🪙K[X]🪙 s)) < ?len p" unfolding Cons s' using a normalize_length_le[of "map2 (⊕) p' s'"] by (auto simp add: univ_poly_def r_neg) thenobtain q' r' where l_div: "?long_division (p ⊕🪙K[X]🪙 (b ⊗🪙K[X]🪙 s)) q' r'" using less(1)[OF _ is_polynomial] by blast
have in_carrier: "p ∈ carrier (K[X])""b ∈ carrier (K[X])""s ∈ carrier (K[X])" "q' ∈ carrier (K[X])""r' ∈ carrier (K[X])" using l_div assms less(2) s unfolding univ_poly_def by auto have"(p ⊕🪙K[X]🪙 (b ⊗🪙K[X]🪙 s)) ⊖🪙K[X]🪙 (b ⊗🪙K[X]🪙 s) = ((b ⊗🪙K[X]🪙 q') ⊕🪙K[X]🪙 r') ⊖🪙K[X]🪙 (b ⊗🪙K[X]🪙 s)" using l_div by simp hence"p = (b ⊗🪙K[X]🪙 (q' ⊖🪙K[X]🪙 s)) ⊕🪙K[X]🪙 r'" using in_carrier by algebra moreoverhave"q' ⊖🪙K[X]🪙 s ∈ carrier (K[X])" using in_carrier by algebra hence"polynomial K (q' ⊖🪙K[X]🪙 s)" unfolding univ_poly_def by simp ultimatelyhave"?long_division p (q' ⊖🪙K[X]🪙 s) r'" using l_div by auto thus ?thesis by blast qed qed qed
end(* of fixed K context. *)
end(* of domain context. *)
(* PROOF ========== *) lemma (indomain) field_long_division_theorem: assumes"subfield K R""polynomial K p"and"polynomial K b""b ≠ []" shows"∃q r. polynomial K q ∧ polynomial K r ∧ p = (b ⊗🪙K[X]🪙 q) ⊕🪙K[X]🪙 r ∧ (r = [] ∨ degree r < degree b)" using long_division_theorem[OF subfieldE(1)[OF assms(1)] assms(2-4)] assms(3-4)
subfield.subfield_Units[OF assms(1)] lead_coeff_not_zero[of K "hd b""tl b"] by simp
(* PROOF ========== *) text‹The same theorem as above, but now, everything is in a shell. › lemma (indomain) field_long_division_theorem_shell: assumes"subfield K R""p ∈ carrier (K[X])"and"b ∈ carrier (K[X])""b ≠0🪙K[X]🪙" shows"∃q r. q ∈ carrier (K[X]) ∧ r ∈ carrier (K[X]) ∧ p = (b ⊗🪙K[X]🪙 q) ⊕🪙K[X]🪙 r ∧ (r = 0🪙K[X]🪙∨ degree r < degree b)" using field_long_division_theorem assms by (auto simp add: univ_poly_def)
subsection‹Consistency Rules›
lemma polynomial_consistent [simp]: shows"polynomial🪙(R ( carrier := K ))🪙 K p ==> polynomial🪙R🪙 K p" unfolding polynomial_def by auto
lemma (in ring) eval_consistent [simp]: assumes"subring K R"shows"ring.eval (R ( carrier := K )) = eval" proof fix p show"ring.eval (R ( carrier := K )) p = eval p" using nat_pow_consistent ring.eval.simps[OF subring_is_ring[OF assms]] by (induct p) (auto) qed
lemma (in ring) coeff_consistent [simp]: assumes"subring K R"shows"ring.coeff (R ( carrier := K )) = coeff" proof fix p show"ring.coeff (R ( carrier := K )) p = coeff p" using ring.coeff.simps[OF subring_is_ring[OF assms]] by (induct p) (auto) qed
lemma (in ring) normalize_consistent [simp]: assumes"subring K R"shows"ring.normalize (R ( carrier := K )) = normalize" proof fix p show"ring.normalize (R ( carrier := K )) p = normalize p" using ring.normalize.simps[OF subring_is_ring[OF assms]] by (induct p) (auto) qed
lemma (in ring) poly_add_consistent [simp]: assumes"subring K R"shows"ring.poly_add (R ( carrier := K )) = poly_add" proof - have"∧p q. ring.poly_add (R ( carrier := K )) p q = poly_add p q" proof - fix p q show"ring.poly_add (R ( carrier := K )) p q = poly_add p q" using ring.poly_add.simps[OF subring_is_ring[OF assms]] normalize_consistent[OF assms] by auto qed thus ?thesis by (auto simp del: poly_add.simps) qed
lemma (in ring) poly_mult_consistent [simp]: assumes"subring K R"shows"ring.poly_mult (R ( carrier := K )) = poly_mult" proof - have"∧p q. ring.poly_mult (R ( carrier := K )) p q = poly_mult p q" proof - fix p q show"ring.poly_mult (R ( carrier := K )) p q = poly_mult p q" using ring.poly_mult.simps[OF subring_is_ring[OF assms]] poly_add_consistent[OF assms] by (induct p) (auto) qed thus ?thesis by auto qed
lemma (indomain) univ_poly_a_inv_consistent: assumes"subring K R""p ∈ carrier (K[X])" shows"⊖🪙K[X]🪙 p = ⊖🪙(carrier R)[X]🪙 p" proof - have in_carrier: "p ∈ carrier ((carrier R)[X])" using assms carrier_polynomial by (auto simp add: univ_poly_def) show ?thesis using univ_poly_a_inv_def'[OF assms]
univ_poly_a_inv_def'[OF carrier_is_subring in_carrier] by simp qed
lemma (indomain) univ_poly_a_minus_consistent: assumes"subring K R""q ∈ carrier (K[X])" shows"p ⊖🪙K[X]🪙 q = p ⊖🪙(carrier R)[X]🪙 q" using univ_poly_a_inv_consistent[OF assms] unfolding a_minus_def univ_poly_def by auto
lemma (in ring) univ_poly_consistent: assumes"subring K R" shows"univ_poly (R ( carrier := K )) = univ_poly R" unfolding univ_poly_def polynomial_def using poly_add_consistent[OF assms]
poly_mult_consistent[OF assms]
subringE(1)[OF assms] by auto
subsubsection ‹Corollaries›
(* PROOF ========== *) corollary (in ring) subfield_long_division_theorem_shell: assumes"subfield K R""p ∈ carrier (K[X])"and"b ∈ carrier (K[X])""b ≠0🪙K[X]🪙" shows"∃q r. q ∈ carrier (K[X]) ∧ r ∈ carrier (K[X]) ∧ p = (b ⊗🪙K[X]🪙 q) ⊕🪙K[X]🪙 r ∧ (r = 0🪙K[X]🪙∨ degree r < degree b)" usingdomain.field_long_division_theorem_shell[OF subdomain_is_domain[OF subfield.axioms(1)]
field.carrier_is_subfield[OF subfield_iff(2)[OF assms(1)]]] assms(1-4) unfolding univ_poly_consistent[OF subfieldE(1)[OF assms(1)]] by auto
corollary (indomain) univ_poly_is_euclidean: assumes"subfield K R"shows"euclidean_domain (K[X]) degree" proof - interpret UP: domain"K[X]" using univ_poly_is_domain[OF subfieldE(1)[OF assms]] field_def by blast show ?thesis using subfield_long_division_theorem_shell[OF assms] by (auto intro!: UP.euclidean_domainI) qed
corollary (indomain) univ_poly_is_principal: assumes"subfield K R"shows"principal_domain (K[X])" proof - interpret UP: euclidean_domain "K[X]" degree using univ_poly_is_euclidean[OF assms] . show ?thesis .. qed
subsection‹The Evaluation Homomorphism›
lemma (in ring) eval_replicate: assumes"set p ⊆ carrier R""a ∈ carrier R" shows"eval ((replicate n 0) @ p) a = eval p a" using assms eval_in_carrier by (induct n) (auto)
lemma (in ring) eval_normalize: assumes"set p ⊆ carrier R""a ∈ carrier R" shows"eval (normalize p) a = eval p a" using eval_replicate[OF normalize_in_carrier] normalize_def'[of p] assms by metis
lemma (in ring) eval_poly_add_aux: assumes"set p ⊆ carrier R""set q ⊆ carrier R"and"length p = length q"and"a ∈ carrier R" shows"eval (poly_add p q) a = (eval p a) ⊕ (eval q a)" proof - have"eval (map2 (⊕) p q) a = (eval p a) ⊕ (eval q a)" using assms proof (induct p arbitrary: q) case Nil thus ?caseby simp next case (Cons b1 p') thenobtain b2 q' where q: "q = b2 # q'" by (metis length_Cons list.exhaust list.size(3) nat.simps(3)) show ?case using eval_in_carrier[OF _ Cons(5), of q']
eval_in_carrier[OF _ Cons(5), of p'] Cons unfolding q by (auto simp add: ring_simprules(7,13,22)) qed moreoverhave"set (map2 (⊕) p q) ⊆ carrier R" using assms(1-2) by (induct p arbitrary: q) (auto, metis add.m_closed in_set_zipE set_ConsD subsetCE) ultimatelyshow ?thesis using assms(3) eval_normalize[OF _ assms(4), of "map2 (⊕) p q"] by auto qed
lemma (in ring) eval_poly_add: assumes"set p ⊆ carrier R""set q ⊆ carrier R"and"a ∈ carrier R" shows"eval (poly_add p q) a = (eval p a) ⊕ (eval q a)" proof - have aux_lemma: "eval (poly_add p q) a = (eval p a) ⊕ (eval q a)" if A: "set p ⊆ carrier R""set q ⊆ carrier R""length p ≥ length q"for p q proof - from that have"eval (poly_add p ((replicate (length p - length q) 0) @ q)) a = (eval p a) ⊕ (eval ((replicate (length p - length q) 0) @ q) a)" using eval_poly_add_aux[OF A(1) _ _ assms(3), of "(replicate (length p - length q) 0) @ q"] by force thenshow"eval (poly_add p q) a = (eval p a) ⊕ (eval q a)" using eval_replicate[OF A(2) assms(3)] A(3) by auto qed have ?thesis if"length q ≥ length p" using assms(1-2)[THEN eval_in_carrier[OF _ assms(3)]] poly_add_comm[OF assms(1-2)]
aux_lemma[OF assms(2,1) that] by (auto simp del: poly_add.simps simp add: add.m_comm) moreoverhave ?thesis if"length p ≥ length q" using aux_lemma[OF assms(1-2) that] . ultimatelyshow ?thesis by auto qed
lemma (in ring) eval_append_aux: assumes"set p ⊆ carrier R"and"b ∈ carrier R"and"a ∈ carrier R" shows"eval (p @ [ b ]) a = ((eval p a) ⊗ a) ⊕ b" using assms(1) proof (induct p) case Nil thus ?caseby (auto simp add: assms(2-3)) next case (Cons l q) have"a [^] length q ∈ carrier R""eval q a ∈ carrier R" using eval_in_carrier Cons(2) assms(2-3) by auto thus ?case using Cons assms(2-3) by (auto, algebra) qed
lemma (in ring) eval_append: assumes"set p ⊆ carrier R""set q ⊆ carrier R"and"a ∈ carrier R" shows"eval (p @ q) a = ((eval p a) ⊗ (a [^] (length q))) ⊕ (eval q a)" using assms(2) proof (induct "length q" arbitrary: q) case 0 thus ?case using eval_in_carrier[OF assms(1,3)] by auto next case (Suc n) thenobtain b q' where q: "q = q' @ [ b ]" by (metis length_Suc_conv list.simps(3) rev_exhaust) hence in_carrier: "eval p a ∈ carrier R""eval q' a ∈ carrier R" "a [^] (length q') ∈ carrier R""b ∈ carrier R" using assms(1,3) Suc(3) eval_in_carrier[OF _ assms(3)] by auto
have"eval (p @ q) a = ((eval (p @ q') a) ⊗ a) ⊕ b" using eval_append_aux[OF _ _ assms(3), of "p @ q'" b] assms(1) Suc(3) unfolding q by auto alsohave" ... = ((((eval p a) ⊗ (a [^] (length q'))) ⊕ (eval q' a)) ⊗ a) ⊕ b" using Suc unfolding q by auto alsohave" ... = (((eval p a) ⊗ ((a [^] (length q')) ⊗ a))) ⊕ (((eval q' a) ⊗ a) ⊕ b)" using assms(3) in_carrier by algebra alsohave" ... = (eval p a) ⊗ (a [^] (length q)) ⊕ (eval q a)" using eval_append_aux[OF _ in_carrier(4) assms(3), of q'] Suc(3) unfolding q by auto finallyshow ?case . qed
lemma (in ring) eval_monom: assumes"b ∈ carrier R"and"a ∈ carrier R" shows"eval (monom b n) a = b ⊗ (a [^] n)" proof (induct n) case 0 thus ?case using assms unfolding monom_def by auto next case (Suc n) have"monom b (Suc n) = (monom b n) @ [ 0 ]" unfolding monom_def by (simp add: replicate_append_same) hence"eval (monom b (Suc n)) a = ((eval (monom b n) a) ⊗ a) ⊕0" using eval_append_aux[OF monom_in_carrier[OF assms(1)] zero_closed assms(2), of n] by simp alsohave" ... = b ⊗ (a [^] (Suc n))" using Suc assms m_assoc by auto finallyshow ?case . qed
lemma (in cring) eval_poly_mult: assumes"set p ⊆ carrier R""set q ⊆ carrier R"and"a ∈ carrier R" shows"eval (poly_mult p q) a = (eval p a) ⊗ (eval q a)" using assms(1) proof (induct p) case Nil thus ?case using eval_in_carrier[OF assms(2-3)] by simp next case (Cons b p)
have aux_lemma: "eval ((map ((⊗) b) q) @ (replicate n 0)) a = (eval (monom b n) a)⊗ (eval q a)" if b: "b ∈ carrier R"for n b proof - from that have"set (map ((⊗) b) q) ⊆ carrier R"and"set (replicate n 0) ⊆ carrier R" using assms(2) by (induct q) (auto) hence"eval ((map ((⊗) b) q) @ (replicate n 0)) a = (eval ((map ((⊗) b) q)) a) ⊗(a [^] n) ⊕0" using eval_append[OF _ _ assms(3), of "map ((⊗) b) q""replicate n 0"]
eval_replicate[OF _ assms(3), of "[]"] by auto moreoverhave"eval (map ((⊗) b) q) a = b ⊗ eval q a" using assms(2-3) eval_in_carrier b by(induct q) (auto simp add: m_assoc r_distr) ultimatelyhave"eval ((map ((⊗) b) q) @ (replicate n 0)) a = (b ⊗ eval q a) ⊗ (a [^] n) ⊕0" by simp alsohave" ... = (b ⊗ (a [^] n)) ⊗ (eval q a)" using eval_in_carrier[OF assms(2-3)] b assms(3) m_assoc m_comm by auto finallyshow ?thesis using eval_monom[OF b assms(3)] by simp qed
from Cons have in_carrier: "eval (monom b (length p)) a ∈ carrier R""eval p a ∈ carrier R""eval q a ∈ carrier R""b ∈ carrier R" using eval_in_carrier monom_in_carrier assms by auto have set_map: "set ((map ((⊗) b) q) @ (replicate (length p) 0)) ⊆ carrier R" using in_carrier(4) assms(2) by (induct q) (auto) have set_poly: "set (poly_mult p q) ⊆ carrier R" using poly_mult_in_carrier[OF _ assms(2), of p] Cons(2) by auto have"eval (poly_mult (b # p) q) a = ((eval (monom b (length p)) a) ⊗ (eval q a)) ⊕ ((eval p a) ⊗ (eval q a))" using eval_poly_add[OF set_map set_poly assms(3)] aux_lemma[OF in_carrier(4), of "length p"] Cons by (auto simp del: poly_add.simps) alsohave" ... = ((eval (monom b (length p)) a) ⊕ (eval p a)) ⊗ (eval q a)" using l_distr[OF in_carrier(1-3)] by simp alsohave" ... = (eval (b # p) a) ⊗ (eval q a)" unfolding eval_monom[OF in_carrier(4) assms(3), of "length p"] by auto finallyshow ?case . qed
proposition (in cring) eval_is_hom: assumes"subring K R"and"a ∈ carrier R" shows"(λp. (eval p) a) ∈ ring_hom (K[X]) R" unfolding univ_poly_def using polynomial_in_carrier[OF assms(1)] eval_in_carrier
eval_poly_add eval_poly_mult assms(2) by (auto intro!: ring_hom_memI
simp add: univ_poly_carrier
simp del: poly_add.simps poly_mult.simps)
theorem (indomain) eval_cring_hom: assumes"subring K R"and"a ∈ carrier R" shows"ring_hom_cring (K[X]) R (λp. (eval p) a)" unfolding ring_hom_cring_def ring_hom_cring_axioms_def usingdomain.axioms(1)[OF univ_poly_is_domain[OF assms(1)]]
eval_is_hom[OF assms] cring_axioms by auto
corollary (indomain) eval_ring_hom: assumes"subring K R"and"a ∈ carrier R" shows"ring_hom_ring (K[X]) R (λp. (eval p) a)" using eval_cring_hom[OF assms] ring_hom_ringI2 unfolding ring_hom_cring_def ring_hom_cring_axioms_def cring_def by auto
subsection‹Homomorphisms›
lemma (in ring_hom_ring) eval_hom': assumes"a ∈ carrier R"and"set p ⊆ carrier R" shows"h (R.eval p a) = eval (map h p) (h a)" using assms by (induct p, auto simp add: R.eval_in_carrier hom_nat_pow)
lemma (in ring_hom_ring) eval_hom: assumes"subring K R"and"a ∈ carrier R"and"p ∈ carrier (K[X])" shows"h (R.eval p a) = eval (map h p) (h a)" proof - have"set p ⊆ carrier R" using subringE(1)[OF assms(1)] R.polynomial_incl assms(3) unfolding sym[OF univ_poly_carrier[of R]] by auto thus ?thesis using eval_hom'[OF assms(2)] by simp qed
lemma (in ring_hom_ring) coeff_hom': assumes"set p ⊆ carrier R"shows"h (R.coeff p i) = coeff (map h p) i" using assms by (induct p) (auto)
lemma (in ring_hom_ring) poly_add_hom': assumes"set p ⊆ carrier R"and"set q ⊆ carrier R" shows"normalize (map h (R.poly_add p q)) = poly_add (map h p) (map h q)" proof - have set_map: "set (map h s) ⊆ carrier S"if"set s ⊆ carrier R"for s using that by auto have"coeff (normalize (map h (R.poly_add p q))) = coeff (map h (R.poly_add p q))" using S.normalize_coeff by auto alsohave" ... = (λi. h ((R.coeff p i) ⊕ (R.coeff q i)))" using coeff_hom'[OF R.poly_add_in_carrier[OF assms]] R.poly_add_coeff[OF assms] by simp alsohave" ... = (λi. (coeff (map h p) i) ⊕🪙S🪙 (coeff (map h q) i))" using assms[THEN R.coeff_in_carrier] assms[THEN coeff_hom'] by simp alsohave" ... = (λi. coeff (poly_add (map h p) (map h q)) i)" using S.poly_add_coeff[OF assms[THEN set_map]] by simp finallyhave"coeff (normalize (map h (R.poly_add p q))) = (λi. coeff (poly_add (map h p) (map h q)) i)" . thus ?thesis unfolding coeff_iff_polynomial_cond[OF
normalize_gives_polynomial[OF set_map[OF R.poly_add_in_carrier[OF assms]]]
poly_add_is_polynomial[OF carrier_is_subring assms[THEN set_map]]] . qed
lemma (in ring_hom_ring) poly_mult_hom': assumes"set p ⊆ carrier R"and"set q ⊆ carrier R" shows"normalize (map h (R.poly_mult p q)) = poly_mult (map h p) (map h q)" using assms(1) proof (induct p, simp) case (Cons a p) have set_map: "set (map h s) ⊆ carrier S"if"set s ⊆ carrier R"for s using that by auto
let ?q_a = "(map ((⊗) a) q) @ (replicate (length p) 0)" have set_q_a: "set ?q_a ⊆ carrier R" using assms(2) Cons(2) by (induct q) (auto) have q_a_simp: "map h ?q_a = (map ((⊗🪙S🪙) (h a)) (map h q)) @ (replicate (length (map h p)) 0🪙S🪙)" using assms(2) Cons(2) by (induct q) (auto)
have"S.normalize (map h (R.poly_mult (a # p) q)) = S.normalize (map h (R.poly_add ?q_a (R.poly_mult p q)))" by simp alsohave" ... = S.poly_add (map h ?q_a) (map h (R.poly_mult p q))" using poly_add_hom'[OF set_q_a R.poly_mult_in_carrier[OF _ assms(2)]] Cons by simp alsohave" ... = S.poly_add (map h ?q_a) (S.normalize (map h (R.poly_mult p q)))" using poly_add_normalize(2)[OF set_map[OF set_q_a] set_map[OF R.poly_mult_in_carrier[OF _ assms(2)]]] Cons by simp alsohave" ... = S.poly_add (map h ?q_a) (S.poly_mult (map h p) (map h q))" using Cons by simp alsohave" ... = S.poly_mult (map h (a # p)) (map h q)" unfolding q_a_simp by simp finallyshow ?case . qed
subsection‹The X Variable›
definition var :: "_ ==> 'a list" (‹X🍋›) where"X🪙R🪙 = [ 1🪙R🪙, 0🪙R🪙 ]"
lemma (in ring) eval_var: assumes"x ∈ carrier R"shows"eval X x = x" using assms unfolding var_def by auto
lemma (indomain) var_closed: assumes"subring K R"shows"X ∈ carrier (K[X])"and"polynomial K X" using subringE(2-3)[OF assms] by (auto simp add: var_def univ_poly_def polynomial_def)
lemma (indomain) poly_mult_var': assumes"set p ⊆ carrier R" shows"poly_mult X p = normalize (p @ [ 0 ])" and"poly_mult p X = normalize (p @ [ 0 ])" proof - from‹set p ⊆ carrier R›have"poly_mult [ 1 ] p = normalize p" using poly_mult_one' by simp thus"poly_mult X p = normalize (p @ [ 0 ])" using poly_mult_append_zero[OF _ assms, of "[ 1 ]"] normalize_idem unfolding var_def by (auto simp del: poly_mult.simps) thus"poly_mult p X = normalize (p @ [ 0 ])" using poly_mult_comm[OF assms] unfolding var_def by simp qed
lemma (indomain) poly_mult_var: assumes"subring K R""p ∈ carrier (K[X])" shows"p ⊗🪙K[X]🪙 X = (if p = [] then [] else p @ [ 0 ])" proof - have is_poly: "polynomial K p" using assms(2) unfolding univ_poly_def by simp hence"polynomial K (p @ [ 0 ])"if"p ≠ []" using that subringE(2)[OF assms(1)] unfolding polynomial_def by auto thus ?thesis using poly_mult_var'(2)[OF polynomial_in_carrier[OF assms(1) is_poly]]
normalize_polynomial[of K "p @ [ 0 ]"] by (auto simp add: univ_poly_mult[of R K]) qed
lemma (indomain) var_pow_closed: assumes"subring K R"shows"X [^]🪙K[X]🪙 (n :: nat) ∈ carrier (K[X])" using monoid.nat_pow_closed[OF univ_poly_is_monoid[OF assms] var_closed(1)[OF assms]] .
lemma (indomain) unitary_monom_eq_var_pow: assumes"subring K R"shows"monom 1 n = X [^]🪙K[X]🪙 n" using poly_mult_var[OF assms var_pow_closed[OF assms]] unfolding nat_pow_def monom_def by (induct n) (auto simp add: univ_poly_one, metis append_Cons replicate_append_same)
lemma (indomain) monom_eq_var_pow: assumes"subring K R""a ∈ carrier R - { 0 }" shows"monom a n = [ a ] ⊗🪙K[X]🪙 (X [^]🪙K[X]🪙 n)" proof - have"monom a n = map ((⊗) a) (monom 1 n)" unfolding monom_def using assms(2) by (induct n) (auto) alsohave" ... = poly_mult [ a ] (monom 1 n)" using poly_mult_const(1)[OF _ monom_is_polynomial assms(2)] carrier_is_subring by simp alsohave" ... = [ a ] ⊗🪙K[X]🪙 (X [^]🪙K[X]🪙 n)" unfolding unitary_monom_eq_var_pow[OF assms(1)] univ_poly_mult[of R K] by simp finallyshow ?thesis . qed
interpret UP: domain"K[X]" using univ_poly_is_domain[OF assms(1)] .
have aux_lemma1: "set (?map_norm l) ⊆ carrier (K[X])"if"set l ⊆ K"for l proof - from that have"poly_of_const a ∈ carrier (K[X])"if"a ∈ set l"for a using that normalize_gives_polynomial[of "[ a ]" K] unfolding univ_poly_carrier poly_of_const_def by auto thenshow ?thesis by auto qed
have aux_lemma2: "UP.eval (?map_norm l) q = UP.eval (?map_norm ((replicate n 0) @ l)) q" if set_l: "set l ⊆ K"and q: "q ∈ carrier (K[X])" for n q l using set_l proof (induct n) case 0 thenshow ?caseby simp next case (Suc n) from‹set l ⊆ K›have set_replicate: "set ((replicate n 0) @ l) ⊆ K" using subringE(2)[OF assms(1)] by (induct n) (auto) have step: "UP.eval (?map_norm l') q = UP.eval (?map_norm (0 # l')) q"if"set l' ⊆K"for l' using UP.eval_in_carrier[OF aux_lemma1[OF that]] q unfolding poly_of_const_def by (simp, simp add: sym[OF univ_poly_zero[of R K]]) have"UP.eval (?map_norm l) q = UP.eval (?map_norm ((replicate n 0) @ l)) q" using Suc by simp alsohave" ... = UP.eval (map poly_of_const ((replicate (Suc n) 0) @ l)) q" using step[OF set_replicate] by simp finallyshow ?case . qed
have aux_lemma3: "UP.eval (?map_norm l) q = UP.eval (?map_norm (normalize l)) q" if"set l ⊆ K"and q: "q ∈ carrier (K[X])"for q l proof - from‹set l ⊆ K›have set_norm: "set (normalize l) ⊆ K" by (induct l) (auto) show ?thesis using aux_lemma2[OF set_norm q, of "length l - length (local.normalize l)"] unfolding sym[OF normalize_trick[of l]] .. qed
from‹p ∈ carrier (K[X])›show ?thesis proof (induct "length p" arbitrary: p rule: less_induct) case less thus ?case proof (cases p) case Nil thenshow ?thesis by (simp add: univ_poly_zero) next case (Cons a l) hence a: "a ∈ carrier R - { 0 }"and set_l: "set l ⊆ carrier R""set l ⊆ K" using less(2) subringE(1)[OF assms(1)] unfolding sym[OF univ_poly_carrier] polynomial_def by auto
have"a # l = poly_add (monom a (length l)) l" using poly_add_monom[OF set_l(1) a] .. alsohave" ... = poly_add (monom a (length l)) (normalize l)" using poly_add_normalize(2)[OF monom_in_carrier[of a] set_l(1)] a by simp alsohave" ... = poly_add (monom a (length l)) (UP.eval (?map_norm (normalize l)) X)" using less(1)[of "normalize l"] normalize_gives_polynomial[OF set_l(2)] normalize_length_le[of l] by (auto simp add: univ_poly_carrier Cons(1)) alsohave" ... = poly_add ([ a ] ⊗🪙K[X]🪙 (X [^]🪙K[X]🪙 (length l))) (UP.eval (?map_norm l) X)" unfolding monom_eq_var_pow[OF assms(1) a] aux_lemma3[OF set_l(2) var_closed(1)[OF assms(1)]] .. alsohave" ... = UP.eval (?map_norm (a # l)) X" using a unfolding sym[OF univ_poly_add[of R K]] unfolding poly_of_const_def by auto finallyshow ?thesis unfolding Cons(1) . qed qed qed
lemma (in ring) dense_repr_set_fst: assumes"set p ⊆ K"shows"fst ` (set (dense_repr p)) ⊆ K - { 0 }" using assms by (induct p) (auto)
lemma (in ring) dense_repr_set_snd: shows"snd ` (set (dense_repr p)) ⊆ {..< length p}" by (induct p) (auto)
lemma (indomain) dense_repr_monom_closed: assumes"subring K R""set p ⊆ K" shows"t ∈ set (dense_repr p) ==> monom (fst t) (snd t) ∈ carrier (K[X])" using dense_repr_set_fst[OF assms(2)] monom_is_polynomial[OF assms(1)] by (auto simp add: univ_poly_carrier)
lemma (indomain) monom_finsum_decomp: assumes"subring K R""p ∈ carrier (K[X])" shows"p = (⨁🪙K[X]🪙 t ∈ set (dense_repr p). monom (fst t) (snd t))" proof - interpret UP: domain"K[X]" using univ_poly_is_domain[OF assms(1)] .
from‹p ∈ carrier (K[X])›show ?thesis proof (induct "length p" arbitrary: p rule: less_induct) case less thus ?case proof (cases p) case Nil thus ?thesis using UP.finsum_empty univ_poly_zero[of R K] by simp next case (Cons a l) hence in_carrier: "normalize l ∈ carrier (K[X])""polynomial K (normalize l)""polynomial K (a # l)" using normalize_gives_polynomial polynomial_incl[of K p] less(2) unfolding univ_poly_carrier by auto have len_lt: "length (local.normalize l) < length p" using normalize_length_le by (simp add: Cons le_imp_less_Suc)
have a: "a ∈ K - { 0 }" using less(2) subringE(1)[OF assms(1)] unfolding Cons univ_poly_def polynomial_def by auto hence"p = (monom a (length l)) ⊕🪙K[X]🪙 (poly_of_dense (dense_repr (normalize l)))" using monom_decomp[OF assms(1), of p] less(2) dense_repr_normalize unfolding univ_poly_add univ_poly_carrier Cons by (auto simp del: poly_add.simps) alsohave" ... = (monom a (length l)) ⊕🪙K[X]🪙 (normalize l)" using monom_decomp[OF assms(1) in_carrier(2)] by simp finallyhave"p = monom a (length l) ⊕🪙K[X]🪙 (⨁🪙K[X]🪙 t ∈ set (dense_repr l). monom (fst t) (snd t))" using less(1)[OF len_lt in_carrier(1)] dense_repr_normalize by simp
moreoverhave"(a, (length l)) ∉ set (dense_repr l)" using dense_repr_set_snd[of l] by auto moreoverhave"monom a (length l) ∈ carrier (K[X])" using monom_is_polynomial[OF assms(1) a] unfolding univ_poly_carrier by simp moreoverhave"∧t. t ∈ set (dense_repr l) ==> monom (fst t) (snd t) ∈ carrier (K[X])" using dense_repr_monom_closed[OF assms(1)] polynomial_incl[OF in_carrier(3)] by auto ultimatelyhave"p = (⨁🪙K[X]🪙 t ∈ set (dense_repr (a # l)). monom (fst t) (snd t))" using UP.add.finprod_insert a by auto thus ?thesis unfolding Cons . qed qed qed
lemma (indomain) var_pow_finsum_decomp: assumes"subring K R""p ∈ carrier (K[X])" shows"p = (⨁🪙K[X]🪙 t ∈ set (dense_repr p). [ fst t ] ⊗🪙K[X]🪙 (X [^]🪙K[X]🪙(snd t)))" proof - let ?f = "λt. monom (fst t) (snd t)" let ?g = "λt. [ fst t ] ⊗🪙K[X]🪙 (X [^]🪙K[X]🪙 (snd t))"
interpret UP: domain"K[X]" using univ_poly_is_domain[OF assms(1)] .
have set_p: "set p ⊆ K" using polynomial_incl assms(2) by (simp add: univ_poly_carrier) hence f: "?f ∈ set (dense_repr p) → carrier (K[X])" using dense_repr_monom_closed[OF assms(1)] by auto
moreover have"∧t. t ∈ set (dense_repr p) ==> fst t ∈ carrier R - { 0 }" using dense_repr_set_fst[OF set_p] subringE(1)[OF assms(1)] by auto hence"∧t. t ∈ set (dense_repr p) ==> monom (fst t) (snd t) = [ fst t ] ⊗🪙K[X]🪙 (X [^]🪙K[X]🪙 (snd t))" using monom_eq_var_pow[OF assms(1)] by auto
ultimatelyshow ?thesis using UP.add.finprod_cong[of _ _ ?f ?g] monom_finsum_decomp[OF assms] by auto qed
corollary (indomain) hom_var_pow_finsum: assumes"subring K R"and"p ∈ carrier (K[X])""ring_hom_ring (K[X]) A h" shows"h p = (⨁🪙A🪙 t ∈ set (dense_repr p). h [ fst t ] ⊗🪙A🪙 (h X [^]🪙A🪙 (snd t)))" proof - let ?f = "λt. [ fst t ] ⊗🪙K[X]🪙 (X [^]🪙K[X]🪙 (snd t))" let ?g = "λt. h [ fst t ] ⊗🪙A🪙 (h X [^]🪙A🪙 (snd t))"
interpret UP: domain"K[X]" + A: ring A using univ_poly_is_domain[OF assms(1)] ring_hom_ring.axioms(2)[OF assms(3)] by simp+
have const_in_carrier: "∧t. t ∈ set (dense_repr p) ==> [ fst t ] ∈ carrier (K[X])" using dense_repr_set_fst[OF polynomial_incl, of K p] assms(2) const_is_polynomial[of _ K] by (auto simp add: univ_poly_carrier) hence f: "?f: set (dense_repr p) → carrier (K[X])" using UP.m_closed[OF _ var_pow_closed[OF assms(1)]] by auto hence h: "h ∘ ?f: set (dense_repr p) → carrier A" using ring_hom_memE(1)[OF ring_hom_ring.homh[OF assms(3)]] by (auto simp add: Pi_def)
have hp: "h p = (⨁🪙A🪙 t ∈ set (dense_repr p). (h ∘ ?f) t)" using ring_hom_ring.hom_finsum[OF assms(3) f] var_pow_finsum_decomp[OF assms(1-2)] by (auto, meson o_apply) have eq: "∧t. t ∈ set (dense_repr p) ==> h [ fst t ] ⊗🪙A🪙 (h X [^]🪙A🪙 (snd t)) = (h ∘ ?f) t" using ring_hom_memE(2)[OF ring_hom_ring.homh[OF assms(3)]
const_in_carrier var_pow_closed[OF assms(1)]]
ring_hom_ring.hom_nat_pow[OF assms(3) var_closed(1)[OF assms(1)]] by auto show ?thesis using A.add.finprod_cong'[OF _ h eq] hp by simp qed
corollary (indomain) determination_of_hom: assumes"subring K R" and"ring_hom_ring (K[X]) A h""ring_hom_ring (K[X]) A g" and"∧k. k ∈ K ==> h [ k ] = g [ k ]"and"h X = g X" shows"∧p. p ∈ carrier (K[X]) ==> h p = g p" proof - interpret A: ring A using ring_hom_ring.axioms(2)[OF assms(2)] by simp
fix p assume p: "p ∈ carrier (K[X])" hence "∧t. t ∈ set (dense_repr p) ==> [ fst t ] ∈ carrier (K[X])" using dense_repr_set_fst[OF polynomial_incl, of K p] const_is_polynomial[of _ K] by (auto simp add: univ_poly_carrier) hence f: "(λt. h [ fst t ] ⊗🪙A🪙 (h X [^]🪙A🪙 (snd t))): set (dense_repr p) → carrier A" using ring_hom_memE(1)[OF ring_hom_ring.homh[OF assms(2)]] var_closed(1)[OF assms(1)]
A.m_closed[OF _ A.nat_pow_closed] by auto
have eq: "∧t. t ∈ set (dense_repr p) ==> g [ fst t ] ⊗🪙A🪙 (g X [^]🪙A🪙 (snd t)) = h [ fst t ] ⊗🪙A🪙 (h X [^]🪙A🪙 (snd t))" using dense_repr_set_fst[OF polynomial_incl, of K p] p assms(4-5) by (auto simp add: univ_poly_carrier) show"h p = g p" unfolding assms(2-3)[THEN hom_var_pow_finsum[OF assms(1) p]] using A.add.finprod_cong'[OF _ f eq] by simp qed
corollary (indomain) eval_as_unique_hom: assumes"subring K R""x ∈ carrier R" and"ring_hom_ring (K[X]) R h" and"∧k. k ∈ K ==> h [ k ] = k"and"h X = x" shows"∧p. p ∈ carrier (K[X]) ==> h p = eval p x" using determination_of_hom[OF assms(1,3) eval_ring_hom[OF assms(1-2)]]
eval_var[OF assms(2)] assms(4-5) subringE(1)[OF assms(1)] by fastforce
subsection‹The Constant Term›
definition (in ring) const_term :: "'a list ==> 'a" where"const_term p = eval p 0"
lemma (in ring) const_term_eq_last: assumes"set p ⊆ carrier R"and"a ∈ carrier R" shows"const_term (p @ [ a ]) = a" using assms by (induct p) (auto simp add: const_term_def)
lemma (in ring) const_term_not_zero: assumes"const_term p ≠0"shows"p ≠ []" using assms by (auto simp add: const_term_def)
lemma (in ring) const_term_explicit: assumes"set p ⊆ carrier R""p ≠ []"and"const_term p = a" obtains p' where"set p' ⊆ carrier R"and"p = p' @ [ a ]" proof - obtain a' p' where p: "p = p' @ [ a' ]" using assms(2) rev_exhaust by blast have p': "set p' ⊆ carrier R"and a: "a = a'" using assms const_term_eq_last[of p' a'] unfolding p by auto show thesis using p p' that unfolding a by blast qed
lemma (in ring) const_term_zero: assumes"subring K R""polynomial K p""p ≠ []"and"const_term p = 0" obtains p' where"polynomial K p'""p' ≠ []"and"p = p' @ [ 0 ]" proof - obtain p' where p': "p = p' @ [ 0 ]" using const_term_explicit[OF polynomial_in_carrier[OF assms(1-2)] assms(3-4)] by auto have"polynomial K p'""p' ≠ []" using assms(2) unfolding p' polynomial_def by auto thus thesis using p' .. qed
lemma (in cring) const_term_simprules: shows"∧p. set p ⊆ carrier R ==> const_term p ∈ carrier R" and"∧p q. [ set p ⊆ carrier R; set q ⊆ carrier R ]==> const_term (poly_mult p q) = const_term p ⊗ const_term q" and"∧p q. [ set p ⊆ carrier R; set q ⊆ carrier R ]==> const_term (poly_add p q) = const_term p ⊕ const_term q" using eval_poly_mult eval_poly_add eval_in_carrier zero_closed unfolding const_term_def by auto
lemma (indomain) const_term_simprules_shell: assumes"subring K R" shows"∧p. p ∈ carrier (K[X]) ==> const_term p ∈ K" and"∧p q. [ p ∈ carrier (K[X]); q ∈ carrier (K[X]) ]==> const_term (p ⊗🪙K[X]🪙 q) = const_term p ⊗ const_term q" and"∧p q. [ p ∈ carrier (K[X]); q ∈ carrier (K[X]) ]==> const_term (p ⊕🪙K[X]🪙 q) = const_term p ⊕ const_term q" and"∧p. p ∈ carrier (K[X]) ==> const_term (⊖🪙K[X]🪙 p) = ⊖ (const_term p)" using eval_is_hom[OF assms(1) zero_closed] unfolding ring_hom_def const_term_def proof (auto) fix p assume p: "p ∈ carrier (K[X])" hence"set p ⊆ carrier R" using polynomial_in_carrier[OF assms(1)] by (auto simp add: univ_poly_def) thus"eval (⊖🪙K [X]🪙 p) 0 = ⊖ local.eval p 0" unfolding univ_poly_a_inv_def'[OF assms(1) p] by (induct p) (auto simp add: eval_in_carrier l_minus local.minus_add)
have"set p ⊆ K" using p by (auto simp add: univ_poly_def polynomial_def) thus"eval p 0∈ K" using subringE(1-2,6-7)[OF assms] by (induct p) (auto, metis assms nat_pow_0 nat_pow_zero subringE(3)) qed
subsection‹The Canonical Embedding of K in K[X]›
lemma (in ring) poly_of_const_consistent: assumes"subring K R"shows"ring.poly_of_const (R ( carrier := K )) = poly_of_const" unfolding ring.poly_of_const_def[OF subring_is_ring[OF assms]]
normalize_consistent[OF assms] poly_of_const_def ..
lemma (indomain) canonical_embedding_is_hom: assumes"subring K R"shows"poly_of_const ∈ ring_hom (R ( carrier := K )) (K[X])" using subringE(1)[OF assms] unfolding subset_iff poly_of_const_def by (auto intro!: ring_hom_memI simp add: univ_poly_def)
lemma (indomain) canonical_embedding_ring_hom: assumes"subring K R"shows"ring_hom_ring (R ( carrier := K )) (K[X]) poly_of_const" using canonical_embedding_is_hom[OF assms] unfolding symmetric[OF ring_hom_ring_axioms_def] by (rule ring_hom_ring.intro[OF subring_is_ring[OF assms] univ_poly_is_ring[OF assms]])
lemma (in field) poly_of_const_over_carrier: shows"poly_of_const ` (carrier R) = { p ∈ carrier ((carrier R)[X]). degree p = 0 }" proof - have"poly_of_const ` (carrier R) = insert [] { [ k ] | k. k ∈ carrier R - { 0 } }" unfolding poly_of_const_def by auto alsohave" ... = { p ∈ carrier ((carrier R)[X]). degree p = 0 }" unfolding univ_poly_def polynomial_def by (auto, metis le_Suc_eq le_zero_eq length_0_conv length_Suc_conv list.sel(1) list.set_sel(1) subsetCE) finallyshow ?thesis . qed
lemma (in ring) poly_of_const_over_subfield: assumes"subfield K R"shows"poly_of_const ` K = { p ∈ carrier (K[X]). degree p = 0 }" using field.poly_of_const_over_carrier[OF subfield_iff(2)[OF assms]]
poly_of_const_consistent[OF subfieldE(1)[OF assms]]
univ_poly_consistent[OF subfieldE(1)[OF assms]] by simp
lemma (in field) univ_poly_carrier_subfield_of_consts: "subfield (poly_of_const ` (carrier R)) ((carrier R)[X])" proof - have ring_hom: "ring_hom_ring R ((carrier R)[X]) poly_of_const" using canonical_embedding_ring_hom[OF carrier_is_subring] by simp thus ?thesis using ring_hom_ring.img_is_subfield(2)[OF ring_hom carrier_is_subfield] unfolding univ_poly_def by auto qed
proposition (in ring) univ_poly_subfield_of_consts: assumes"subfield K R"shows"subfield (poly_of_const ` K) (K[X])" using field.univ_poly_carrier_subfield_of_consts[OF subfield_iff(2)[OF assms]] unfolding poly_of_const_consistent[OF subfieldE(1)[OF assms]]
univ_poly_consistent[OF subfieldE(1)[OF assms]] by simp
end
Messung V0.5 in Prozent
¤ Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.0.79Bemerkung:
(vorverarbeitet am 2026-05-03)
¤
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.