Theory Reverse_Symmetry
theory Reverse_Symmetry
imports Main
begin
chapter "Reverse symmetry"
text ‹This theory deals with a mechanism which produces new facts on lists from already known facts
by the reverse symmetry of lists, induced by the mapping @{term rev}.
It constructs the rule attribute ``reversed'' which produces the symmetrical fact using so-called
reversal rules, which are rewriting rules that may be applied to obtain the symmetrical fact.
An example of such a reversal rule is the already existing @{thm rev_append[symmetric, no_vars]}.
Some additional reversal rules are given in this theory.
The symmetrical fact 'A[reversed]' is constructed from the fact 'A' in the following manner:
1. each schematic variable @{term "xs::'a list"} of type @{typ "'a list"}
is instantiated by @{term "rev xs"};
2. constant Nil is substituted by @{term "rev Nil"};
3. each quantification of @{typ "'a list"} type variable @{term "⋀(xs::'a list). P xs"}
is substituted by (logically equivalent) quantification @{term "⋀xs. P (rev xs)"},
similarly for $\forall$, $\exists$ and $\exists!$ quantifiers; each bounded quantification of @{typ "'a list"} type
variable @{term "∀(xs::'a list) ∈ A. P xs"} is substituted by (logically equivalent)
quantification @{term "∀xs∈rev ` A. P (rev xs)"}, similarly for bounded $\exists$ quantifier;
4. simultaneous rewrites according to a the current list of reversal rules are performed;
5. final correctional rewrites related to reversion of @{const "Cons"} are performed.
List of reversal rules is maintained by declaration attribute ``reversal\_rule'' with standard
``add'' and ``del'' options.
See examples at the end of the file.
›
section ‹Quantifications and maps›
lemma all_surj_conv: assumes "surj f" shows "(⋀x. PROP P (f x)) ≡ (⋀y. PROP P y)"
proof
fix y assume "⋀x. PROP P (f x)"
then have "PROP P (f (inv f y))".
then show "PROP P y" unfolding surj_f_inv_f[OF assms].
qed
lemma All_surj_conv: assumes "surj f" shows "(∀x. P (f x)) ⟷ (∀y. P y)"
proof (intro iffI allI)
fix y assume "∀x. P (f x)"
then have "P (f (inv f y))"..
then show "P y" unfolding surj_f_inv_f[OF assms].
qed simp
lemma Ex_surj_conv: assumes "surj f" shows "(∃x. P (f x)) ⟷ (∃y. P y)"
proof
assume "∃x. P (f x)"
then obtain x where "P (f x)"..
then show "∃x. P x"..
next
assume "∃y. P y"
then obtain y where "P y"..
then have "P (f (inv f y))" unfolding surj_f_inv_f[OF assms].
then show "∃x. P (f x)"..
qed
lemma Ex1_bij_conv: assumes "bij f" shows "(∃!x. P (f x)) ⟷ (∃!y. P y)"
proof
have imp: "∃!y. Q y" if bij: "bij g" and ex1: "∃!x. Q (g x)" for g Q
proof -
from ex1E[OF ex1, rule_format]
obtain x where ex: "Q (g x)" and uniq: "⋀x'. Q (g x') ⟹ x' = x" by blast
{
fix y assume "Q y"
then have "Q (g (inv g y))" unfolding surj_f_inv_f[OF bij_is_surj[OF bij]].
from uniq[OF this] have "x = inv g y"..
then have "y = g x" unfolding bij_inv_eq_iff[OF bij]..
}
with ex show "∃!y. Q y"..
qed
show "∃!x. P (f x) ⟹ ∃!y. P y" using imp[OF assms].
assume "∃!y. P y"
then have "∃!y. P (f (inv f y))" unfolding surj_f_inv_f[OF bij_is_surj[OF assms]].
from imp[OF bij_imp_bij_inv[OF assms] this]
show "∃!x. P (f x)".
qed
lemma Ball_inj_conv: assumes "inj f" shows "(∀y∈f ` A. P (inv f y)) ⟷ (∀x∈A. P x)"
using ball_simps(9)[of f A "λy. P (inv f y)"] unfolding inv_f_f[OF assms].
lemma Bex_inj_conv: assumes "inj f" shows "(∃y∈f ` A. P (inv f y)) ⟷ (∃x∈A. P x)"
using bex_simps(7)[of f A "λy. P (inv f y)"] unfolding inv_f_f[OF assms].
subsection ‹Quantifications and reverse›
lemma rev_involution': "rev ∘ rev = id"
by auto
lemma rev_inv: "inv rev = rev"
using inv_unique_comp[OF rev_involution' rev_involution'].
section ‹Attributes›
context
begin
subsection ‹Cons reversion›
definition snocs :: "'a list ⇒ 'a list ⇒ 'a list"
where "snocs xs ys = xs @ ys"
subsection ‹Final corrections›
lemma snocs_snocs: "snocs (snocs xs (y # ys)) zs = snocs xs (y # snocs ys zs)"
unfolding snocs_def by simp
lemma snocs_Nil: "snocs [] xs = xs"
unfolding snocs_def by simp
lemma snocs_is_append: "snocs xs ys = xs @ ys"
unfolding snocs_def..
private lemmas final_correct1 =
snocs_snocs
private lemmas final_correct2 =
snocs_Nil
private lemmas final_correct3 =
snocs_is_append
subsection ‹Declaration attribute ‹reversal_rule››
ML ‹
structure Reversal_Rules =
Named_Thms(
val name = @{binding "reversal_rule"}
val description = "Rules performing reverse-symmetry transformation on theorems on lists"
)
›
attribute_setup reversal_rule =
‹Attrib.add_del
(Thm.declaration_attribute Reversal_Rules.add_thm)
(Thm.declaration_attribute Reversal_Rules.del_thm)›
"maintaining a list of reversal rules"
subsection ‹Tracing attribute›
ML ‹
val reversed_trace = Config.declare_bool ("reversed_trace", ⌂) (K false);
val enable_tracing = Config.put_generic reversed_trace true
val tracing_attr = Thm.declaration_attribute (K enable_tracing)
val tracing_parser : attribute context_parser = Scan.lift (Scan.succeed tracing_attr)
›
attribute_setup reversed_trace = tracing_parser "reversed trace configuration"
subsection ‹Rule attribute ‹reversed››
private lemma rev_Nil: "rev [] ≡ []"
by simp
private lemma map_Nil: "map f [] ≡ []"
by simp
private lemma image_empty: "f ` Set.empty ≡ Set.empty"
by simp
definition COMP :: "('b ⇒ prop) ⇒ ('a ⇒ 'b) ⇒ 'a ⇒ prop" (infixl "oo" 55)
where "F oo g ≡ (λx. F (g x))"
lemma COMP_assoc: "F oo (f o g) ≡ (F oo f) oo g"
unfolding COMP_def o_def.
private lemma image_comp_image: "(`) f ∘ (`) g ≡ (`) (f ∘ g)"
unfolding comp_def image_comp.
private lemma rev_involution: "rev ∘ rev ≡ id"
unfolding comp_def rev_rev_ident id_def.
private lemma map_involution: assumes "f ∘ f ≡ id" shows "(map f) ∘ (map f) ≡ id"
unfolding map_comp_map ‹f ∘ f ≡ id› List.map.id.
private lemma image_involution: assumes "f ∘ f ≡ id" shows "(image f) ∘ (image f) ≡ id"
unfolding image_comp_image ‹f ∘ f ≡ id› image_id.
private lemma rev_map_comm: "rev ∘ map f ≡ map f ∘ rev"
unfolding comp_def rev_map.
private lemma involut_comm_comp: assumes "f o f ≡ id" and "g o g ≡ id" and "f o g ≡ g o f"
shows "(f ∘ g) ∘ (f ∘ g) ≡ id"
by (simp add: comp_assoc comp_assoc[symmetric] assms)
private lemma rev_map_involution: assumes "g o g ≡ id"
shows "(rev ∘ map g) ∘ (rev ∘ map g) ≡ id"
using involut_comm_comp[OF rev_involution map_involution[OF ‹g o g ≡ id›] rev_map_comm].
private lemma prop_abs_subst: assumes "f o f ≡ id" shows "(λx. F (f x)) oo f ≡ (λx. F x)"
unfolding COMP_def o_apply[symmetric] unfolding ‹f o f ≡ id› id_def.
private lemma prop_abs_subst_comm: assumes "f o f ≡ id" and "g o g ≡ id" and "f o g ≡ g o f"
shows "(λx. F (f (g x))) oo (f o g) ≡ (λx. F x)"
unfolding ‹f o g ≡ g o f› unfolding COMP_assoc
unfolding prop_abs_subst[OF ‹g o g ≡ id›, of "λx. F (f x)"] prop_abs_subst[OF ‹f o f ≡ id›].
private lemma prop_abs_subst_rev_map: assumes "g o g ≡ id"
shows "(λx. F (rev (map g x))) oo (rev o map g) ≡ (λx. F x)"
using prop_abs_subst_comm[OF rev_involution map_involution[OF ‹g o g ≡ id›] rev_map_comm].
private lemma obj_abs_subst: assumes "f o f ≡ id" shows "(λx. F (f x)) o f ≡ (λx. F x)"
unfolding comp_def unfolding o_apply[of f, symmetric] ‹f o f ≡ id› id_def.
private lemma obj_abs_subst_comm: assumes "f o f ≡ id" and "g o g ≡ id" and "f o g ≡ g o f"
shows "(λx. F (f (g x))) o (f o g) ≡ (λx. F x)"
unfolding ‹f o g ≡ g o f› unfolding comp_assoc[symmetric]
unfolding obj_abs_subst[OF ‹g o g ≡ id›, of "λx. F (f x)"] obj_abs_subst[OF ‹f o f ≡ id›].
private lemma obj_abs_subst_rev_map: assumes "g o g ≡ id"
shows "(λx. F (rev (map g x))) o (rev o map g) ≡ (λx. F x)"
using obj_abs_subst_comm[OF rev_involution map_involution[OF ‹g o g ≡ id›] rev_map_comm].
ML ‹
local
fun comp_const T = Const(\<^const_name>‹comp›, (T --> T) --> (T --> T) --> T --> T)
fun rev_const T = Const(\<^const_name>‹rev›, T --> T)
fun map_const S T = Const(\<^const_name>‹map›, (S --> S) --> T --> T)
fun image_const S T = Const(\<^const_name>‹image›, (S --> S) --> T --> T)
val rev_Nil_thm = @{thm rev_Nil}
val map_Nil_thm = @{thm map_Nil}
val image_empty_thm = @{thm image_empty}
val rev_involut_thm = @{thm rev_involution}
val map_involut_thm = @{thm map_involution}
val image_involut_thm = @{thm image_involution}
val rev_map_comm_thm = @{thm rev_map_comm}
val involut_comm_comp_thm = @{thm involut_comm_comp}
fun abs_subst_thm T =
if T = propT then @{thm prop_abs_subst} else @{thm obj_abs_subst}
fun abs_subst_rev_map_thm T =
if T = propT then @{thm prop_abs_subst_rev_map} else @{thm obj_abs_subst_rev_map}
fun comp T f gs = fold (fn f => fn g =>
(comp_const T $ f $ g)) gs f
fun app ctxt gs ct = fold_rev (fn g => fn ct' =>
Thm.apply (Thm.cterm_of ctxt g) ct') gs ct
in
fun subst ctxt T ct =
let
fun tr (T as Type(\<^type_name>‹list›, [S])) = [rev_const T] @ (
case tr S of
[] => []
| (g :: gs') => [map_const S T $ comp S g gs'])
| tr (T as Type(\<^type_name>‹set›, [S])) = (
case tr S of
[] => []
| (g :: gs') => [image_const S T $ comp S g gs'])
| tr _ = []
in app ctxt (tr T) ct end
fun abs_cv T U ct =
let
fun tr_eq (T as Type(\<^type_name>‹list›, [S])) =
rev_involut_thm :: (
case tr_eq S of
[] => []
| [f_eq] => [f_eq RS map_involut_thm]
| [f_eq, g_eq] =>
[([f_eq, g_eq, rev_map_comm_thm] MRS involut_comm_comp_thm) RS map_involut_thm])
| tr_eq (T as Type(\<^type_name>‹set›, [S])) = (
case tr_eq S of
[] => []
| [f_eq] => [f_eq RS image_involut_thm]
| [f_eq, g_eq] =>
[([f_eq, g_eq, rev_map_comm_thm] MRS involut_comm_comp_thm) RS image_involut_thm])
| tr_eq _ = []
in case tr_eq T of
[] => Thm.reflexive ct
| [f_inv] =>
[Thm.reflexive ct, Thm.symmetric (f_inv RS abs_subst_thm U)] MRS transitive_thm
| [f_inv, g_inv] =>
[Thm.reflexive ct, Thm.symmetric (g_inv RS abs_subst_rev_map_thm U)] MRS transitive_thm
end;
fun Nil_cv ctxt T ct =
((Conv.try_conv o Conv.arg_conv o Conv.rewr_conv) map_Nil_thm
then_conv (Conv.try_conv o Conv.rewr_conv) rev_Nil_thm) (subst ctxt T ct)
|> Thm.symmetric
fun empty_cv ctxt T ct =
(Conv.try_conv o Conv.rewr_conv) image_empty_thm (subst ctxt T ct)
|> Thm.symmetric
end
fun initiate_cv ctxt ct =
case Thm.term_of ct of
_ $ _ => Conv.comb_conv (initiate_cv ctxt) ct
| Abs(_, T, b) => (Conv.abs_conv (initiate_cv o snd) ctxt then_conv abs_cv T (fastype_of b)) ct
| Const(\<^const_name>‹Nil›, T) => Nil_cv ctxt T ct
| Const(\<^const_name>‹bot›, T as Type(\<^type_name>‹set›, _)) => empty_cv ctxt T ct
| _ => Thm.reflexive ct
›
ML ‹
fun trace_rule_prems_proof ctxt rule goals rule_prems rule' =
if not (Config.get ctxt reversed_trace) then () else
let
val ctxt_prems = Raw_Simplifier.prems_of ctxt
val np = Thm.nprems_of rule
val np' = Thm.nprems_of rule'
val pretty_term = Syntax.pretty_term ctxt
val pretty_thm = pretty_term o Thm.prop_of
val success = rule_prems |> List.all is_some
val sendback = Active.make_markup Markup.sendbackN
{implicit = false, properties = [Markup.padding_command]}
val _ = [
[
"Trying to use conditional reverse rule:" |> Pretty.para,
rule |> pretty_thm
] |> Pretty.fbreaks |> Pretty.block,
[(if null ctxt_prems
then "No context premises."
else "Context premises:"
) |> Pretty.para
] @ (
ctxt_prems |> map (Pretty.item o single o pretty_thm)
) |> Pretty.fbreaks |> Pretty.block,
( if success then [
"Successfully derived unconditional reverse rule:" |> Pretty.para,
rule' |> pretty_thm
] else [
"Unable to prove " ^ string_of_int np ^ " out of " ^ string_of_int np' ^ " rule premises:\n"
|> Pretty.para
] @ (
(goals ~~ rule_prems) |> map_filter (
fn (goal, NONE) => SOME ([
"lemma" |> Pretty.str, Pretty.brk 1,
goal |> pretty_term |> Pretty.quote, Pretty.fbrk,
"sorry" |> Pretty.str
] |> curry Pretty.blk 0 |> Pretty.mark sendback |> single |> Pretty.item)
| _ => NONE
)
)) |> Pretty.fbreaks |> Pretty.block
] |> Pretty.chunks |> Pretty.string_of |> tracing
in () end
fun full_resolve ctxt prem i =
let
val tac = resolve_tac ctxt [prem] THEN_ALL_NEW blast_tac ctxt
in rule_by_tactic ctxt (tac i) end
fun prover method ss ctxt rule =
let
val ctxt_prems = Raw_Simplifier.prems_of ctxt
val rule_prems' = Logic.strip_imp_prems (Thm.prop_of rule)
val goals = rule_prems' |> map (fn prem =>
Logic.list_implies (map Thm.prop_of ctxt_prems, prem));
val ctxt' = ctxt |> put_simpset ss
fun prove t = SOME (Goal.prove ctxt' [] [] t
(fn {context = goal_ctxt, prems} => NO_CONTEXT_TACTIC goal_ctxt (method goal_ctxt prems)))
handle ERROR _ => NONE
val ths = goals |> map prove
val gen_ctxt_prems = map (Variable.gen_all ctxt) ctxt_prems
fun full_resolve1 prem = full_resolve ctxt prem 1
val rule_prems = ths |> (map o Option.map) (fold full_resolve1 gen_ctxt_prems)
val rule' = (fold o curry) (
fn (SOME th, rule') => rule' |> full_resolve1 th
| (NONE, rule') => Drule.rotate_prems 1 rule'
) rule_prems rule
val nprems = Thm.nprems_of rule'
val _ = trace_rule_prems_proof ctxt rule goals rule_prems rule'
in if nprems = 0 then SOME rule' else NONE end
fun rewrite _ _ [] = Thm.reflexive
| rewrite method ctxt thms =
let
val p = prover method (simpset_of ctxt)
val ctxt' = Raw_Simplifier.init_simpset thms ctxt
in
Raw_Simplifier.rewrite_cterm (true, true, true) p ctxt'
end
fun rewrite_rule method ctxt = Conv.fconv_rule o rewrite method ctxt;
fun meta_reversal_rules ctxt extra =
map (Local_Defs.meta_rewrite_rule ctxt) (extra @ Reversal_Rules.get ctxt)
fun reverse method extra_rules context th =
let
val ctxt = Context.proof_of context
val final_correct1 = map (Local_Defs.meta_rewrite_rule ctxt) @{thms final_correct1}
val final_correct2 = map (Local_Defs.meta_rewrite_rule ctxt) @{thms final_correct2}
val final_correct3 = map (Local_Defs.meta_rewrite_rule ctxt) @{thms final_correct3}
val rules = meta_reversal_rules ctxt extra_rules
val cvars = Thm.add_vars th Vars.empty
val cvars' = Vars.map ((subst ctxt o snd)) cvars
val th_subst = Thm.instantiate (TVars.empty, cvars') th
val ((_, [th_import]), ctxt') = Variable.import true [th_subst] ctxt
val th_init = th_import |> Conv.fconv_rule (initiate_cv ctxt')
val th_rev = th_init |> rewrite_rule method ctxt' rules
val th_corr = th_rev
|> Raw_Simplifier.rewrite_rule ctxt' final_correct1
|> Raw_Simplifier.rewrite_rule ctxt' final_correct2
|> Raw_Simplifier.rewrite_rule ctxt' final_correct3
val th_export = th_corr |> singleton (Variable.export ctxt' ctxt)
in
Drule.zero_var_indexes th_export
end
val default_method = SIMPLE_METHOD o CHANGED_PROP o auto_tac
val solve_arg = Args.$$$ "solve"
val extra_rules_parser = Scan.optional (Scan.lift (Args.add -- Args.colon) |-- Attrib.thms) []
val solve_parser = Scan.lift (Scan.optional
(solve_arg -- Args.colon |-- Method.parse >> (fst #> Method.evaluate)) default_method
)
val reversed = extra_rules_parser -- solve_parser
>> (fn (ths, method) => Thm.rule_attribute [] (reverse method ths))
›
attribute_setup reversed =
‹reversed›
"Transforms the theorem on lists to reverse-symmetric version"
end
section ‹Declaration of basic reversal rules›
subsection ‹Pure›
lemma all_surj_conv' [reversal_rule]: assumes "surj f" shows "Pure.all (P oo f) ≡ Pure.all P"
unfolding COMP_def using all_surj_conv[OF assms].
subsection ‹\<^theory>‹HOL.HOL››
lemmas [reversal_rule] = rev_is_rev_conv inj_eq
lemma All_surj_conv' [reversal_rule]: assumes "surj f" shows "All (P o f) = All P"
unfolding comp_def using All_surj_conv[OF assms].
lemma Ex_surj_conv' [reversal_rule]: assumes "surj f" shows "Ex (P o f) ⟷ Ex P"
unfolding comp_def using Ex_surj_conv[OF assms].
lemma Ex1_bij_conv' [reversal_rule]: assumes "bij f" shows "Ex1 (P o f) ⟷ Ex1 P"
unfolding comp_def using Ex1_bij_conv[OF assms].
lemma if_image [reversal_rule]: "(if P then f u else f v) = f (if P then u else v)"
by simp
subsection ‹\<^theory>‹HOL.Set››
lemma collect_image: "Collect (P ∘ f) = f -` (Collect P)"
by fastforce
lemma collect_image' [reversal_rule]: assumes "f ∘ f = id" shows "Collect (P ∘ f) = f ` (Collect P)"
unfolding collect_image
unfolding bij_vimage_eq_inv_image[OF o_bij[OF assms assms]]
unfolding inv_unique_comp[OF assms assms]..
lemma Ball_image [reversal_rule]: assumes "(g ∘ f) ` A = A" shows "Ball (f ` A) (P ∘ g) = Ball A P"
unfolding Ball_image_comp[symmetric] image_comp ‹(g ∘ f) ` A = A›..
lemma Bex_image_comp: "Bex (f ` A) g = Bex A (g ∘ f)"
by simp
lemma Bex_image [reversal_rule]: assumes "(g ∘ f) ` A = A" shows "Bex (f ` A) (P ∘ g) = Bex A P"
unfolding Bex_image_comp[symmetric] image_comp ‹(g ∘ f) ` A = A›..
lemma insert_image [reversal_rule]: "insert (f x) (f ` X) = f ` (insert x X)"
by blast
lemmas [reversal_rule] = inj_image_mem_iff
lemmas [reversal_rule] = inj_image_subset_iff
subsection ‹\<^theory>‹HOL.List››
lemmas [reversal_rule] = set_rev set_map
lemma Cons_rev: "a # rev u = rev (snocs u [a])"
unfolding snocs_def by simp
lemma Cons_map: "(f x) # (map f xs) = map f (x # xs)"
using list.simps(9)[symmetric].
lemmas [reversal_rule] = Cons_rev Cons_map
lemmas [reversal_rule] = hd_rev hd_map
lemma tl_rev: "tl (rev xs) = rev (butlast xs)"
using butlast_rev[of "rev xs", symmetric] unfolding rev_swap rev_rev_ident.
lemmas [reversal_rule] = tl_rev map_tl[symmetric]
lemmas [reversal_rule] = last_rev last_map
lemmas [reversal_rule] = butlast_rev map_butlast[symmetric]
lemma coset_rev: "List.coset (rev xs) = List.coset xs"
by simp
lemma coset_map: assumes "bij f" shows "List.coset (map f xs) = f ` List.coset xs"
using bij_image_Compl_eq[OF ‹bij f›, symmetric] unfolding coset_def set_map.
lemmas [reversal_rule] = coset_rev coset_map
lemmas [reversal_rule] = rev_append[symmetric] map_append[symmetric]
lemma concat_rev_map_rev: "concat (rev (map rev xs)) = rev (concat xs)"
using rev_concat[symmetric] unfolding rev_map.
lemma concat_rev_map_rev': "concat (rev (map (rev ∘ f) xs)) = rev (concat (map f xs))"
unfolding map_comp_map[symmetric] o_apply using concat_rev_map_rev.
lemmas [reversal_rule] = concat_rev_map_rev concat_rev_map_rev'
lemmas [reversal_rule] = drop_rev drop_map
lemmas [reversal_rule] = take_rev take_map
lemmas [reversal_rule] = rev_nth nth_map
lemma list_insert_map [reversal_rule]:
assumes "inj f" shows "List.insert (f x) (map f xs) = map f (List.insert x xs)"
unfolding List.insert_def set_map inj_image_mem_iff[OF ‹inj f›] Cons_map if_image..
lemma list_union_map [reversal_rule]:
assumes "inj f" shows "List.union (map f xs) (map f ys) = map f (List.union xs ys)"
proof (induction xs arbitrary: ys)
case (Cons a xs)
show ?case using Cons.IH unfolding List.union_def Cons_map[symmetric] fold.simps(2) o_apply
unfolding list_insert_map[OF ‹inj f›].
qed (simp add: List.union_def)
lemmas [reversal_rule] = length_rev length_map
lemmas [reversal_rule] = rotate_rev rotate_map
lemma rev_in_lists: "rev u ∈ lists A ⟷ u ∈ lists A"
by auto
lemma map_in_lists: "inj f ⟹ map f u ∈ lists (f ` A) ⟷ u ∈ lists A"
by (simp add: lists_image inj_image_mem_iff inj_mapI)
lemmas [reversal_rule] = rev_in_lists map_in_lists
lemmas [reversal_rule] = list_all_rev
lemmas [reversal_rule] = list_ex_rev
subsection ‹Reverse Symmetry›
lemma snocs_map [reversal_rule]: "snocs (map f u) [f a] = map f (snocs u [a])"
unfolding snocs_def by simp
section ‹›
lemma bij_rev: "bij rev"
using o_bij[OF rev_involution' rev_involution'].
lemma bij_map: "bij f ⟹ bij (map f)"
using bij_betw_def inj_mapI lists_UNIV lists_image by metis
lemma surj_map: "surj f ⟹ surj (map f)"
using lists_UNIV lists_image by metis
lemma bij_image: "bij f ⟹ bij (image f)"
using bij_betw_Pow by force
lemma inj_image: "inj f ⟹ inj (image f)"
by (simp add: inj_on_image)
lemma surj_image: "surj f ⟹ surj (image f)"
using Pow_UNIV image_Pow_surj by metis
lemmas [simp] =
bij_rev
bij_is_inj
bij_is_surj
bij_comp
inj_compose
comp_surj
bij_map
inj_mapI
surj_map
bij_image
inj_image
surj_image
section ‹Examples›
context
begin
subsection ‹Cons and append›
private lemma example_Cons_append:
assumes "xs = [a, b]" and "ys = [b, a, b]"
shows "xs @ xs @ xs = a # b # a # ys"
using assms by simp
thm
example_Cons_append
example_Cons_append[reversed]
example_Cons_append[reversed, reversed]
thm
not_Cons_self
not_Cons_self[reversed]
thm
neq_Nil_conv
neq_Nil_conv[reversed]
subsection ‹Induction rules›
thm
list_nonempty_induct
list_nonempty_induct[reversed] list_nonempty_induct[reversed, where P="λx. P (rev x)" for P, unfolded rev_rev_ident]
thm
list_induct2
list_induct2[reversed] list_induct2[reversed, where P="λx y. P (rev x) (rev y)" for P, unfolded rev_rev_ident]
subsection ‹hd, tl, last, butlast›
thm
hd_append
hd_append[reversed]
last_append
thm
length_tl
length_tl[reversed]
length_butlast
thm
hd_Cons_tl
hd_Cons_tl[reversed]
append_butlast_last_id
append_butlast_last_id[reversed]
subsection ‹set›
thm
hd_in_set
hd_in_set[reversed]
last_in_set
thm
set_ConsD
set_ConsD[reversed]
thm
split_list_first
split_list_first[reversed]
thm
split_list_first_prop
split_list_first_prop[reversed]
split_list_first_prop[reversed, unfolded append_assoc append_Cons append_Nil]
split_list_last_prop
thm
split_list_first_propE
split_list_first_propE[reversed]
split_list_first_propE[reversed, unfolded append_assoc append_Cons append_Nil]
split_list_last_propE
subsection ‹rotate›
private lemma rotate1_hd_tl: "xs ≠ [] ⟹ rotate 1 xs = tl xs @ [hd xs]"
by (cases xs) simp_all
thm
rotate1_hd_tl
rotate1_hd_tl[reversed]
end
end