src/HOL/Factorial.thy
author wenzelm
Thu, 08 Apr 2021 16:43:35 +0200
changeset 73547 a7aabdf889b7
parent 72569 d56e4eeae967
child 74969 fa0020b47868
permissions -rw-r--r--
clarified signature;
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
66589
b884c42694e0 tuned headers;
wenzelm
parents: 66394
diff changeset
     1
(*  Title:      HOL/Factorial.thy
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
     2
    Author:     Jacques D. Fleuriot
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
     3
    Author:     Lawrence C Paulson
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
     4
    Author:     Jeremy Avigad
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
     5
    Author:     Chaitanya Mangla
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
     6
    Author:     Manuel Eberl
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
     7
*)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
     8
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
     9
section \<open>Factorial Function, Rising Factorials\<close>
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    10
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    11
theory Factorial
65813
bdd17b18e103 relaxed theory dependencies
haftmann
parents: 65812
diff changeset
    12
  imports Groups_List
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    13
begin
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    14
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    15
subsection \<open>Factorial Function\<close>
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    16
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    17
context semiring_char_0
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    18
begin
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    19
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    20
definition fact :: "nat \<Rightarrow> 'a"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    21
  where fact_prod: "fact n = of_nat (\<Prod>{1..n})"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    22
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    23
lemma fact_prod_Suc: "fact n = of_nat (prod Suc {0..<n})"
70113
c8deb8ba6d05 Fixing the main Homology theory; also moving a lot of sum/prod lemmas into their generic context
paulson <lp15@cam.ac.uk>
parents: 69182
diff changeset
    24
  unfolding fact_prod using atLeast0LessThan prod.atLeast1_atMost_eq by auto
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    25
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    26
lemma fact_prod_rev: "fact n = of_nat (\<Prod>i = 0..<n. n - i)"
70113
c8deb8ba6d05 Fixing the main Homology theory; also moving a lot of sum/prod lemmas into their generic context
paulson <lp15@cam.ac.uk>
parents: 69182
diff changeset
    27
proof -
c8deb8ba6d05 Fixing the main Homology theory; also moving a lot of sum/prod lemmas into their generic context
paulson <lp15@cam.ac.uk>
parents: 69182
diff changeset
    28
  have "prod Suc {0..<n} = \<Prod>{1..n}"
c8deb8ba6d05 Fixing the main Homology theory; also moving a lot of sum/prod lemmas into their generic context
paulson <lp15@cam.ac.uk>
parents: 69182
diff changeset
    29
    by (simp add: atLeast0LessThan prod.atLeast1_atMost_eq)
c8deb8ba6d05 Fixing the main Homology theory; also moving a lot of sum/prod lemmas into their generic context
paulson <lp15@cam.ac.uk>
parents: 69182
diff changeset
    30
  then have "prod Suc {0..<n} = prod ((-) (n + 1)) {1..n}"
c8deb8ba6d05 Fixing the main Homology theory; also moving a lot of sum/prod lemmas into their generic context
paulson <lp15@cam.ac.uk>
parents: 69182
diff changeset
    31
    using prod.atLeastAtMost_rev [of "\<lambda>i. i" 1 n] by presburger
c8deb8ba6d05 Fixing the main Homology theory; also moving a lot of sum/prod lemmas into their generic context
paulson <lp15@cam.ac.uk>
parents: 69182
diff changeset
    32
  then show ?thesis
c8deb8ba6d05 Fixing the main Homology theory; also moving a lot of sum/prod lemmas into their generic context
paulson <lp15@cam.ac.uk>
parents: 69182
diff changeset
    33
    unfolding fact_prod_Suc by (simp add: atLeast0LessThan prod.atLeast1_atMost_eq)
c8deb8ba6d05 Fixing the main Homology theory; also moving a lot of sum/prod lemmas into their generic context
paulson <lp15@cam.ac.uk>
parents: 69182
diff changeset
    34
qed
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    35
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    36
lemma fact_0 [simp]: "fact 0 = 1"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    37
  by (simp add: fact_prod)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    38
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    39
lemma fact_1 [simp]: "fact 1 = 1"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    40
  by (simp add: fact_prod)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    41
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    42
lemma fact_Suc_0 [simp]: "fact (Suc 0) = 1"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    43
  by (simp add: fact_prod)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    44
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    45
lemma fact_Suc [simp]: "fact (Suc n) = of_nat (Suc n) * fact n"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    46
  by (simp add: fact_prod atLeastAtMostSuc_conv algebra_simps)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    47
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    48
lemma fact_2 [simp]: "fact 2 = 2"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    49
  by (simp add: numeral_2_eq_2)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    50
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    51
lemma fact_split: "k \<le> n \<Longrightarrow> fact n = of_nat (prod Suc {n - k..<n}) * fact (n - k)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    52
  by (simp add: fact_prod_Suc prod.union_disjoint [symmetric]
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    53
    ivl_disj_un ac_simps of_nat_mult [symmetric])
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    54
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    55
end
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    56
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    57
lemma of_nat_fact [simp]: "of_nat (fact n) = fact n"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    58
  by (simp add: fact_prod)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    59
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    60
lemma of_int_fact [simp]: "of_int (fact n) = fact n"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    61
  by (simp only: fact_prod of_int_of_nat_eq)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    62
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    63
lemma fact_reduce: "n > 0 \<Longrightarrow> fact n = of_nat n * fact (n - 1)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    64
  by (cases n) auto
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    65
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    66
lemma fact_nonzero [simp]: "fact n \<noteq> (0::'a::{semiring_char_0,semiring_no_zero_divisors})"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    67
  apply (induct n)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    68
  apply auto
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    69
  using of_nat_eq_0_iff
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    70
  apply fastforce
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    71
  done
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    72
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    73
lemma fact_mono_nat: "m \<le> n \<Longrightarrow> fact m \<le> (fact n :: nat)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    74
  by (induct n) (auto simp: le_Suc_eq)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    75
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    76
lemma fact_in_Nats: "fact n \<in> \<nat>"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    77
  by (induct n) auto
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    78
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    79
lemma fact_in_Ints: "fact n \<in> \<int>"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    80
  by (induct n) auto
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    81
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    82
context
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    83
  assumes "SORT_CONSTRAINT('a::linordered_semidom)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    84
begin
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    85
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    86
lemma fact_mono: "m \<le> n \<Longrightarrow> fact m \<le> (fact n :: 'a)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    87
  by (metis of_nat_fact of_nat_le_iff fact_mono_nat)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    88
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    89
lemma fact_ge_1 [simp]: "fact n \<ge> (1 :: 'a)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    90
  by (metis le0 fact_0 fact_mono)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    91
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    92
lemma fact_gt_zero [simp]: "fact n > (0 :: 'a)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    93
  using fact_ge_1 less_le_trans zero_less_one by blast
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    94
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    95
lemma fact_ge_zero [simp]: "fact n \<ge> (0 :: 'a)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    96
  by (simp add: less_imp_le)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    97
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    98
lemma fact_not_neg [simp]: "\<not> fact n < (0 :: 'a)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
    99
  by (simp add: not_less_iff_gr_or_eq)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   100
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   101
lemma fact_le_power: "fact n \<le> (of_nat (n^n) :: 'a)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   102
proof (induct n)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   103
  case 0
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   104
  then show ?case by simp
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   105
next
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   106
  case (Suc n)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   107
  then have *: "fact n \<le> (of_nat (Suc n ^ n) ::'a)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   108
    by (rule order_trans) (simp add: power_mono del: of_nat_power)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   109
  have "fact (Suc n) = (of_nat (Suc n) * fact n ::'a)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   110
    by (simp add: algebra_simps)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   111
  also have "\<dots> \<le> of_nat (Suc n) * of_nat (Suc n ^ n)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   112
    by (simp add: * ordered_comm_semiring_class.comm_mult_left_mono del: of_nat_power)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   113
  also have "\<dots> \<le> of_nat (Suc n ^ Suc n)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   114
    by (metis of_nat_mult order_refl power_Suc)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   115
  finally show ?case .
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   116
qed
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   117
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   118
end
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   119
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   120
lemma fact_less_mono_nat: "0 < m \<Longrightarrow> m < n \<Longrightarrow> fact m < (fact n :: nat)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   121
  by (induct n) (auto simp: less_Suc_eq)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   122
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   123
lemma fact_less_mono: "0 < m \<Longrightarrow> m < n \<Longrightarrow> fact m < (fact n :: 'a::linordered_semidom)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   124
  by (metis of_nat_fact of_nat_less_iff fact_less_mono_nat)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   125
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   126
lemma fact_ge_Suc_0_nat [simp]: "fact n \<ge> Suc 0"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   127
  by (metis One_nat_def fact_ge_1)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   128
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   129
lemma dvd_fact: "1 \<le> m \<Longrightarrow> m \<le> n \<Longrightarrow> m dvd fact n"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   130
  by (induct n) (auto simp: dvdI le_Suc_eq)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   131
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   132
lemma fact_ge_self: "fact n \<ge> n"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   133
  by (cases "n = 0") (simp_all add: dvd_imp_le dvd_fact)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   134
66806
a4e82b58d833 abolished (semi)ring_div in favour of euclidean_(semi)ring_cancel
haftmann
parents: 66589
diff changeset
   135
lemma fact_dvd: "n \<le> m \<Longrightarrow> fact n dvd (fact m :: 'a::linordered_semidom)"
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   136
  by (induct m) (auto simp: le_Suc_eq)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   137
66806
a4e82b58d833 abolished (semi)ring_div in favour of euclidean_(semi)ring_cancel
haftmann
parents: 66589
diff changeset
   138
lemma fact_mod: "m \<le> n \<Longrightarrow> fact n mod (fact m :: 'a::{semidom_modulo, linordered_semidom}) = 0"
a4e82b58d833 abolished (semi)ring_div in favour of euclidean_(semi)ring_cancel
haftmann
parents: 66589
diff changeset
   139
  by (simp add: mod_eq_0_iff_dvd fact_dvd)
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   140
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   141
lemma fact_div_fact:
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   142
  assumes "m \<ge> n"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   143
  shows "fact m div fact n = \<Prod>{n + 1..m}"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   144
proof -
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   145
  obtain d where "d = m - n"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   146
    by auto
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   147
  with assms have "m = n + d"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   148
    by auto
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   149
  have "fact (n + d) div (fact n) = \<Prod>{n + 1..n + d}"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   150
  proof (induct d)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   151
    case 0
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   152
    show ?case by simp
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   153
  next
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   154
    case (Suc d')
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   155
    have "fact (n + Suc d') div fact n = Suc (n + d') * fact (n + d') div fact n"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   156
      by simp
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   157
    also from Suc.hyps have "\<dots> = Suc (n + d') * \<Prod>{n + 1..n + d'}"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   158
      unfolding div_mult1_eq[of _ "fact (n + d')"] by (simp add: fact_mod)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   159
    also have "\<dots> = \<Prod>{n + 1..n + Suc d'}"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   160
      by (simp add: atLeastAtMostSuc_conv)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   161
    finally show ?case .
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   162
  qed
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   163
  with \<open>m = n + d\<close> show ?thesis by simp
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   164
qed
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   165
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   166
lemma fact_num_eq_if: "fact m = (if m = 0 then 1 else of_nat m * fact (m - 1))"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   167
  by (cases m) auto
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   168
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   169
lemma fact_div_fact_le_pow:
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   170
  assumes "r \<le> n"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   171
  shows "fact n div fact (n - r) \<le> n ^ r"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   172
proof -
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   173
  have "r \<le> n \<Longrightarrow> \<Prod>{n - r..n} = (n - r) * \<Prod>{Suc (n - r)..n}" for r
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   174
    by (subst prod.insert[symmetric]) (auto simp: atLeastAtMost_insertL)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   175
  with assms show ?thesis
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   176
    by (induct r rule: nat.induct) (auto simp add: fact_div_fact Suc_diff_Suc mult_le_mono)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   177
qed
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   178
68783
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   179
lemma prod_Suc_fact: "prod Suc {0..<n} = fact n"
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   180
  by (simp add: fact_prod_Suc)
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   181
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   182
lemma prod_Suc_Suc_fact: "prod Suc {Suc 0..<n} = fact n"
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   183
proof (cases "n = 0")
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   184
  case True
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   185
  then show ?thesis by simp
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   186
next
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   187
  case False
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   188
  have "prod Suc {Suc 0..<n} = Suc 0 * prod Suc {Suc 0..<n}"
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   189
    by simp
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   190
  also have "\<dots> = prod Suc (insert 0 {Suc 0..<n})"
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   191
    by simp
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   192
  also have "insert 0 {Suc 0..<n} = {0..<n}"
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   193
    using False by auto
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   194
  finally show ?thesis
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   195
    by (simp add: fact_prod_Suc)
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   196
qed
248e1678ce55 more theorems on fact
haftmann
parents: 68224
diff changeset
   197
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   198
lemma fact_numeral: "fact (numeral k) = numeral k * fact (pred_numeral k)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   199
  \<comment> \<open>Evaluation for specific numerals\<close>
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   200
  by (metis fact_Suc numeral_eq_Suc of_nat_numeral)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   201
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   202
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   203
subsection \<open>Pochhammer's symbol: generalized rising factorial\<close>
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   204
68224
1f7308050349 prefer HTTPS;
wenzelm
parents: 67969
diff changeset
   205
text \<open>See \<^url>\<open>https://en.wikipedia.org/wiki/Pochhammer_symbol\<close>.\<close>
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   206
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   207
context comm_semiring_1
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   208
begin
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   209
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   210
definition pochhammer :: "'a \<Rightarrow> nat \<Rightarrow> 'a"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   211
  where pochhammer_prod: "pochhammer a n = prod (\<lambda>i. a + of_nat i) {0..<n}"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   212
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   213
lemma pochhammer_prod_rev: "pochhammer a n = prod (\<lambda>i. a + of_nat (n - i)) {1..n}"
67411
3f4b0c84630f restored naming of lemmas after corresponding constants
haftmann
parents: 67399
diff changeset
   214
  using prod.atLeastLessThan_rev_at_least_Suc_atMost [of "\<lambda>i. a + of_nat i" 0 n]
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   215
  by (simp add: pochhammer_prod)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   216
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   217
lemma pochhammer_Suc_prod: "pochhammer a (Suc n) = prod (\<lambda>i. a + of_nat i) {0..n}"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   218
  by (simp add: pochhammer_prod atLeastLessThanSuc_atLeastAtMost)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   219
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   220
lemma pochhammer_Suc_prod_rev: "pochhammer a (Suc n) = prod (\<lambda>i. a + of_nat (n - i)) {0..n}"
70113
c8deb8ba6d05 Fixing the main Homology theory; also moving a lot of sum/prod lemmas into their generic context
paulson <lp15@cam.ac.uk>
parents: 69182
diff changeset
   221
  using prod.atLeast_Suc_atMost_Suc_shift
c8deb8ba6d05 Fixing the main Homology theory; also moving a lot of sum/prod lemmas into their generic context
paulson <lp15@cam.ac.uk>
parents: 69182
diff changeset
   222
  by (simp add: pochhammer_prod_rev prod.atLeast_Suc_atMost_Suc_shift del: prod.cl_ivl_Suc)
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   223
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   224
lemma pochhammer_0 [simp]: "pochhammer a 0 = 1"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   225
  by (simp add: pochhammer_prod)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   226
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   227
lemma pochhammer_1 [simp]: "pochhammer a 1 = a"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   228
  by (simp add: pochhammer_prod lessThan_Suc)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   229
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   230
lemma pochhammer_Suc0 [simp]: "pochhammer a (Suc 0) = a"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   231
  by (simp add: pochhammer_prod lessThan_Suc)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   232
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   233
lemma pochhammer_Suc: "pochhammer a (Suc n) = pochhammer a n * (a + of_nat n)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   234
  by (simp add: pochhammer_prod atLeast0_lessThan_Suc ac_simps)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   235
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   236
end
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   237
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   238
lemma pochhammer_nonneg:
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   239
  fixes x :: "'a :: linordered_semidom"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   240
  shows "x > 0 \<Longrightarrow> pochhammer x n \<ge> 0"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   241
  by (induction n) (auto simp: pochhammer_Suc intro!: mult_nonneg_nonneg add_nonneg_nonneg)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   242
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   243
lemma pochhammer_pos:
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   244
  fixes x :: "'a :: linordered_semidom"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   245
  shows "x > 0 \<Longrightarrow> pochhammer x n > 0"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   246
  by (induction n) (auto simp: pochhammer_Suc intro!: mult_pos_pos add_pos_nonneg)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   247
69182
2424301cc73d more and generalized lemmas
haftmann
parents: 69064
diff changeset
   248
context comm_semiring_1
2424301cc73d more and generalized lemmas
haftmann
parents: 69064
diff changeset
   249
begin
2424301cc73d more and generalized lemmas
haftmann
parents: 69064
diff changeset
   250
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   251
lemma pochhammer_of_nat: "pochhammer (of_nat x) n = of_nat (pochhammer x n)"
69182
2424301cc73d more and generalized lemmas
haftmann
parents: 69064
diff changeset
   252
  by (simp add: pochhammer_prod Factorial.pochhammer_prod)
2424301cc73d more and generalized lemmas
haftmann
parents: 69064
diff changeset
   253
2424301cc73d more and generalized lemmas
haftmann
parents: 69064
diff changeset
   254
end
2424301cc73d more and generalized lemmas
haftmann
parents: 69064
diff changeset
   255
2424301cc73d more and generalized lemmas
haftmann
parents: 69064
diff changeset
   256
context comm_ring_1
2424301cc73d more and generalized lemmas
haftmann
parents: 69064
diff changeset
   257
begin
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   258
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   259
lemma pochhammer_of_int: "pochhammer (of_int x) n = of_int (pochhammer x n)"
69182
2424301cc73d more and generalized lemmas
haftmann
parents: 69064
diff changeset
   260
  by (simp add: pochhammer_prod Factorial.pochhammer_prod)
2424301cc73d more and generalized lemmas
haftmann
parents: 69064
diff changeset
   261
2424301cc73d more and generalized lemmas
haftmann
parents: 69064
diff changeset
   262
end
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   263
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   264
lemma pochhammer_rec: "pochhammer a (Suc n) = a * pochhammer (a + 1) n"
70113
c8deb8ba6d05 Fixing the main Homology theory; also moving a lot of sum/prod lemmas into their generic context
paulson <lp15@cam.ac.uk>
parents: 69182
diff changeset
   265
  by (simp add: pochhammer_prod prod.atLeast0_lessThan_Suc_shift ac_simps del: prod.op_ivl_Suc)
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   266
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   267
lemma pochhammer_rec': "pochhammer z (Suc n) = (z + of_nat n) * pochhammer z n"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   268
  by (simp add: pochhammer_prod prod.atLeast0_lessThan_Suc ac_simps)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   269
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   270
lemma pochhammer_fact: "fact n = pochhammer 1 n"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   271
  by (simp add: pochhammer_prod fact_prod_Suc)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   272
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   273
lemma pochhammer_of_nat_eq_0_lemma: "k > n \<Longrightarrow> pochhammer (- (of_nat n :: 'a:: idom)) k = 0"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   274
  by (auto simp add: pochhammer_prod)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   275
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   276
lemma pochhammer_of_nat_eq_0_lemma':
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   277
  assumes kn: "k \<le> n"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   278
  shows "pochhammer (- (of_nat n :: 'a::{idom,ring_char_0})) k \<noteq> 0"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   279
proof (cases k)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   280
  case 0
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   281
  then show ?thesis by simp
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   282
next
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   283
  case (Suc h)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   284
  then show ?thesis
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   285
    apply (simp add: pochhammer_Suc_prod)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   286
    using Suc kn
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   287
    apply (auto simp add: algebra_simps)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   288
    done
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   289
qed
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   290
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   291
lemma pochhammer_of_nat_eq_0_iff:
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   292
  "pochhammer (- (of_nat n :: 'a::{idom,ring_char_0})) k = 0 \<longleftrightarrow> k > n"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   293
  (is "?l = ?r")
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   294
  using pochhammer_of_nat_eq_0_lemma[of n k, where ?'a='a]
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   295
    pochhammer_of_nat_eq_0_lemma'[of k n, where ?'a = 'a]
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   296
  by (auto simp add: not_le[symmetric])
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   297
66394
32084d7e6b59 Some facts about orders of zeros
eberlm <eberlm@in.tum.de>
parents: 65813
diff changeset
   298
lemma pochhammer_0_left:
32084d7e6b59 Some facts about orders of zeros
eberlm <eberlm@in.tum.de>
parents: 65813
diff changeset
   299
  "pochhammer 0 n = (if n = 0 then 1 else 0)"
32084d7e6b59 Some facts about orders of zeros
eberlm <eberlm@in.tum.de>
parents: 65813
diff changeset
   300
  by (induction n) (simp_all add: pochhammer_rec)
32084d7e6b59 Some facts about orders of zeros
eberlm <eberlm@in.tum.de>
parents: 65813
diff changeset
   301
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   302
lemma pochhammer_eq_0_iff: "pochhammer a n = (0::'a::field_char_0) \<longleftrightarrow> (\<exists>k < n. a = - of_nat k)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   303
  by (auto simp add: pochhammer_prod eq_neg_iff_add_eq_0)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   304
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   305
lemma pochhammer_eq_0_mono:
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   306
  "pochhammer a n = (0::'a::field_char_0) \<Longrightarrow> m \<ge> n \<Longrightarrow> pochhammer a m = 0"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   307
  unfolding pochhammer_eq_0_iff by auto
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   308
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   309
lemma pochhammer_neq_0_mono:
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   310
  "pochhammer a m \<noteq> (0::'a::field_char_0) \<Longrightarrow> m \<ge> n \<Longrightarrow> pochhammer a n \<noteq> 0"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   311
  unfolding pochhammer_eq_0_iff by auto
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   312
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   313
lemma pochhammer_minus:
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   314
  "pochhammer (- b) k = ((- 1) ^ k :: 'a::comm_ring_1) * pochhammer (b - of_nat k + 1) k"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   315
proof (cases k)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   316
  case 0
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   317
  then show ?thesis by simp
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   318
next
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   319
  case (Suc h)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   320
  have eq: "((- 1) ^ Suc h :: 'a) = (\<Prod>i = 0..h. - 1)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   321
    using prod_constant [where A="{0.. h}" and y="- 1 :: 'a"]
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   322
    by auto
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   323
  with Suc show ?thesis
67969
83c8cafdebe8 Syntax for the special cases Min(A`I) and Max (A`I)
paulson <lp15@cam.ac.uk>
parents: 67411
diff changeset
   324
    using pochhammer_Suc_prod_rev [of "b - of_nat k + 1"] 
83c8cafdebe8 Syntax for the special cases Min(A`I) and Max (A`I)
paulson <lp15@cam.ac.uk>
parents: 67411
diff changeset
   325
    by (auto simp add: pochhammer_Suc_prod prod.distrib [symmetric] eq of_nat_diff simp del: prod_constant)
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   326
qed
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   327
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   328
lemma pochhammer_minus':
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   329
  "pochhammer (b - of_nat k + 1) k = ((- 1) ^ k :: 'a::comm_ring_1) * pochhammer (- b) k"
67969
83c8cafdebe8 Syntax for the special cases Min(A`I) and Max (A`I)
paulson <lp15@cam.ac.uk>
parents: 67411
diff changeset
   330
  by (simp add: pochhammer_minus)
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   331
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   332
lemma pochhammer_same: "pochhammer (- of_nat n) n =
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   333
    ((- 1) ^ n :: 'a::{semiring_char_0,comm_ring_1,semiring_no_zero_divisors}) * fact n"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   334
  unfolding pochhammer_minus
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   335
  by (simp add: of_nat_diff pochhammer_fact)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   336
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   337
lemma pochhammer_product': "pochhammer z (n + m) = pochhammer z n * pochhammer (z + of_nat n) m"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   338
proof (induct n arbitrary: z)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   339
  case 0
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   340
  then show ?case by simp
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   341
next
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   342
  case (Suc n z)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   343
  have "pochhammer z (Suc n) * pochhammer (z + of_nat (Suc n)) m =
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   344
      z * (pochhammer (z + 1) n * pochhammer (z + 1 + of_nat n) m)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   345
    by (simp add: pochhammer_rec ac_simps)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   346
  also note Suc[symmetric]
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   347
  also have "z * pochhammer (z + 1) (n + m) = pochhammer z (Suc (n + m))"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   348
    by (subst pochhammer_rec) simp
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   349
  finally show ?case
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   350
    by simp
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   351
qed
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   352
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   353
lemma pochhammer_product:
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   354
  "m \<le> n \<Longrightarrow> pochhammer z n = pochhammer z m * pochhammer (z + of_nat m) (n - m)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   355
  using pochhammer_product'[of z m "n - m"] by simp
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   356
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   357
lemma pochhammer_times_pochhammer_half:
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   358
  fixes z :: "'a::field_char_0"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   359
  shows "pochhammer z (Suc n) * pochhammer (z + 1/2) (Suc n) = (\<Prod>k=0..2*n+1. z + of_nat k / 2)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   360
proof (induct n)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   361
  case 0
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   362
  then show ?case
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   363
    by (simp add: atLeast0_atMost_Suc)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   364
next
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   365
  case (Suc n)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   366
  define n' where "n' = Suc n"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   367
  have "pochhammer z (Suc n') * pochhammer (z + 1 / 2) (Suc n') =
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   368
      (pochhammer z n' * pochhammer (z + 1 / 2) n') * ((z + of_nat n') * (z + 1/2 + of_nat n'))"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   369
    (is "_ = _ * ?A")
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   370
    by (simp_all add: pochhammer_rec' mult_ac)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   371
  also have "?A = (z + of_nat (Suc (2 * n + 1)) / 2) * (z + of_nat (Suc (Suc (2 * n + 1))) / 2)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   372
    (is "_ = ?B")
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   373
    by (simp add: field_simps n'_def)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   374
  also note Suc[folded n'_def]
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   375
  also have "(\<Prod>k=0..2 * n + 1. z + of_nat k / 2) * ?B = (\<Prod>k=0..2 * Suc n + 1. z + of_nat k / 2)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   376
    by (simp add: atLeast0_atMost_Suc)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   377
  finally show ?case
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   378
    by (simp add: n'_def)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   379
qed
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   380
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   381
lemma pochhammer_double:
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   382
  fixes z :: "'a::field_char_0"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   383
  shows "pochhammer (2 * z) (2 * n) = of_nat (2^(2*n)) * pochhammer z n * pochhammer (z+1/2) n"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   384
proof (induct n)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   385
  case 0
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   386
  then show ?case by simp
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   387
next
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   388
  case (Suc n)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   389
  have "pochhammer (2 * z) (2 * (Suc n)) = pochhammer (2 * z) (2 * n) *
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   390
      (2 * (z + of_nat n)) * (2 * (z + of_nat n) + 1)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   391
    by (simp add: pochhammer_rec' ac_simps)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   392
  also note Suc
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   393
  also have "of_nat (2 ^ (2 * n)) * pochhammer z n * pochhammer (z + 1/2) n *
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   394
        (2 * (z + of_nat n)) * (2 * (z + of_nat n) + 1) =
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   395
      of_nat (2 ^ (2 * (Suc n))) * pochhammer z (Suc n) * pochhammer (z + 1/2) (Suc n)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   396
    by (simp add: field_simps pochhammer_rec')
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   397
  finally show ?case .
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   398
qed
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   399
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   400
lemma fact_double:
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   401
  "fact (2 * n) = (2 ^ (2 * n) * pochhammer (1 / 2) n * fact n :: 'a::field_char_0)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   402
  using pochhammer_double[of "1/2::'a" n] by (simp add: pochhammer_fact)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   403
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   404
lemma pochhammer_absorb_comp: "(r - of_nat k) * pochhammer (- r) k = r * pochhammer (-r + 1) k"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   405
  (is "?lhs = ?rhs")
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   406
  for r :: "'a::comm_ring_1"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   407
proof -
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   408
  have "?lhs = - pochhammer (- r) (Suc k)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   409
    by (subst pochhammer_rec') (simp add: algebra_simps)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   410
  also have "\<dots> = ?rhs"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   411
    by (subst pochhammer_rec) simp
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   412
  finally show ?thesis .
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   413
qed
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   414
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   415
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   416
subsection \<open>Misc\<close>
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   417
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   418
lemma fact_code [code]:
69064
5840724b1d71 Prefix form of infix with * on either side no longer needs special treatment
nipkow
parents: 68783
diff changeset
   419
  "fact n = (of_nat (fold_atLeastAtMost_nat ((*)) 2 n 1) :: 'a::semiring_char_0)"
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   420
proof -
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   421
  have "fact n = (of_nat (\<Prod>{1..n}) :: 'a)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   422
    by (simp add: fact_prod)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   423
  also have "\<Prod>{1..n} = \<Prod>{2..n}"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   424
    by (intro prod.mono_neutral_right) auto
69064
5840724b1d71 Prefix form of infix with * on either side no longer needs special treatment
nipkow
parents: 68783
diff changeset
   425
  also have "\<dots> = fold_atLeastAtMost_nat ((*)) 2 n 1"
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   426
    by (simp add: prod_atLeastAtMost_code)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   427
  finally show ?thesis .
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   428
qed
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   429
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   430
lemma pochhammer_code [code]:
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   431
  "pochhammer a n =
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   432
    (if n = 0 then 1
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   433
     else fold_atLeastAtMost_nat (\<lambda>n acc. (a + of_nat n) * acc) 0 (n - 1) 1)"
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   434
  by (cases n)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   435
    (simp_all add: pochhammer_prod prod_atLeastAtMost_code [symmetric]
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   436
      atLeastLessThanSuc_atLeastAtMost)
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   437
72569
d56e4eeae967 mult_le_cancel_iff1, mult_le_cancel_iff2, mult_less_iff1 generalised from the real_ versions
paulson <lp15@cam.ac.uk>
parents: 70113
diff changeset
   438
d56e4eeae967 mult_le_cancel_iff1, mult_le_cancel_iff2, mult_less_iff1 generalised from the real_ versions
paulson <lp15@cam.ac.uk>
parents: 70113
diff changeset
   439
lemma mult_less_iff1: "0 < z \<Longrightarrow> x * z < y * z \<longleftrightarrow> x < y"
d56e4eeae967 mult_le_cancel_iff1, mult_le_cancel_iff2, mult_less_iff1 generalised from the real_ versions
paulson <lp15@cam.ac.uk>
parents: 70113
diff changeset
   440
  for x y z :: "'a::linordered_idom"
d56e4eeae967 mult_le_cancel_iff1, mult_le_cancel_iff2, mult_less_iff1 generalised from the real_ versions
paulson <lp15@cam.ac.uk>
parents: 70113
diff changeset
   441
  by simp 
d56e4eeae967 mult_le_cancel_iff1, mult_le_cancel_iff2, mult_less_iff1 generalised from the real_ versions
paulson <lp15@cam.ac.uk>
parents: 70113
diff changeset
   442
d56e4eeae967 mult_le_cancel_iff1, mult_le_cancel_iff2, mult_less_iff1 generalised from the real_ versions
paulson <lp15@cam.ac.uk>
parents: 70113
diff changeset
   443
lemma mult_le_cancel_iff1: "0 < z \<Longrightarrow> x * z \<le> y * z \<longleftrightarrow> x \<le> y"
d56e4eeae967 mult_le_cancel_iff1, mult_le_cancel_iff2, mult_less_iff1 generalised from the real_ versions
paulson <lp15@cam.ac.uk>
parents: 70113
diff changeset
   444
  for x y z :: "'a::linordered_idom"
d56e4eeae967 mult_le_cancel_iff1, mult_le_cancel_iff2, mult_less_iff1 generalised from the real_ versions
paulson <lp15@cam.ac.uk>
parents: 70113
diff changeset
   445
  by simp
d56e4eeae967 mult_le_cancel_iff1, mult_le_cancel_iff2, mult_less_iff1 generalised from the real_ versions
paulson <lp15@cam.ac.uk>
parents: 70113
diff changeset
   446
d56e4eeae967 mult_le_cancel_iff1, mult_le_cancel_iff2, mult_less_iff1 generalised from the real_ versions
paulson <lp15@cam.ac.uk>
parents: 70113
diff changeset
   447
lemma mult_le_cancel_iff2: "0 < z \<Longrightarrow> z * x \<le> z * y \<longleftrightarrow> x \<le> y"
d56e4eeae967 mult_le_cancel_iff1, mult_le_cancel_iff2, mult_less_iff1 generalised from the real_ versions
paulson <lp15@cam.ac.uk>
parents: 70113
diff changeset
   448
  for x y z :: "'a::linordered_idom"
d56e4eeae967 mult_le_cancel_iff1, mult_le_cancel_iff2, mult_less_iff1 generalised from the real_ versions
paulson <lp15@cam.ac.uk>
parents: 70113
diff changeset
   449
  by simp 
d56e4eeae967 mult_le_cancel_iff1, mult_le_cancel_iff2, mult_less_iff1 generalised from the real_ versions
paulson <lp15@cam.ac.uk>
parents: 70113
diff changeset
   450
65812
04ba6d530c87 explicit theory for factorials
haftmann
parents:
diff changeset
   451
end