1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
#[cfg(test)]
extern crate num_bigint;
extern crate num_traits;
use num_traits::{CheckedMul, Signed, Unsigned};
use std::ops::RangeInclusive;
pub trait Factorial<Target = Self> {
fn checked_factorial(&self) -> Option<Target>;
fn factorial(&self) -> Target {
self.checked_factorial()
.expect("Overflow computing factorial")
}
}
impl<T: PartialOrd + Unsigned + CheckedMul> Factorial<T> for T {
#[inline(always)]
fn checked_factorial(&self) -> Option<T> {
let mut acc = T::one();
let mut i = T::one() + T::one();
while i <= *self {
if let Some(acc_i) = acc.checked_mul(&i) {
acc = acc_i;
i = i + T::one();
} else {
return None;
}
}
Some(acc)
}
}
#[cfg(test)]
mod tests {
use super::*;
use num_bigint::*;
#[test]
fn zero_fact_is_one() {
assert_eq!(0u32.factorial(), 1u32);
}
#[test]
fn one_fact_is_one() {
assert_eq!(1.factorial(), 1u32);
}
#[test]
fn two_fact_is_two() {
assert_eq!(2.factorial(), 2u32);
}
#[test]
fn ten_fact() {
assert_eq!(10u32.factorial(), 3628800);
}
#[test]
#[should_panic(expected = "Overflow computing factorial")]
fn too_large() {
100u32.factorial();
}
#[test]
fn too_large_safe() {
assert_eq!(100u32.checked_factorial(), None)
}
#[test]
fn biguint_support() {
assert_eq!(
2u32.to_biguint().unwrap().factorial(),
2u32.to_biguint().unwrap()
);
}
}