bigint.rs 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. #![feature(test)]
  2. extern crate test;
  3. extern crate num;
  4. extern crate rand;
  5. use std::mem::replace;
  6. use test::Bencher;
  7. use num::{BigInt, BigUint, Zero, One, FromPrimitive};
  8. use num::bigint::RandBigInt;
  9. use rand::{SeedableRng, StdRng};
  10. fn get_rng() -> StdRng {
  11. let seed: &[_] = &[1, 2, 3, 4];
  12. SeedableRng::from_seed(seed)
  13. }
  14. fn multiply_bench(b: &mut Bencher, xbits: usize, ybits: usize) {
  15. let mut rng = get_rng();
  16. let x = rng.gen_bigint(xbits);
  17. let y = rng.gen_bigint(ybits);
  18. b.iter(|| &x * &y);
  19. }
  20. fn divide_bench(b: &mut Bencher, xbits: usize, ybits: usize) {
  21. let mut rng = get_rng();
  22. let x = rng.gen_bigint(xbits);
  23. let y = rng.gen_bigint(ybits);
  24. b.iter(|| &x / &y);
  25. }
  26. fn factorial(n: usize) -> BigUint {
  27. let mut f: BigUint = One::one();
  28. for i in 1..(n+1) {
  29. let bu: BigUint = FromPrimitive::from_usize(i).unwrap();
  30. f = f * bu;
  31. }
  32. f
  33. }
  34. fn fib(n: usize) -> BigUint {
  35. let mut f0: BigUint = Zero::zero();
  36. let mut f1: BigUint = One::one();
  37. for _ in 0..n {
  38. let f2 = f0 + &f1;
  39. f0 = replace(&mut f1, f2);
  40. }
  41. f0
  42. }
  43. #[bench]
  44. fn multiply_0(b: &mut Bencher) {
  45. multiply_bench(b, 1 << 8, 1 << 8);
  46. }
  47. #[bench]
  48. fn multiply_1(b: &mut Bencher) {
  49. multiply_bench(b, 1 << 8, 1 << 16);
  50. }
  51. #[bench]
  52. fn multiply_2(b: &mut Bencher) {
  53. multiply_bench(b, 1 << 16, 1 << 16);
  54. }
  55. #[bench]
  56. fn divide_0(b: &mut Bencher) {
  57. divide_bench(b, 1 << 8, 1 << 6);
  58. }
  59. #[bench]
  60. fn divide_1(b: &mut Bencher) {
  61. divide_bench(b, 1 << 12, 1 << 8);
  62. }
  63. #[bench]
  64. fn divide_2(b: &mut Bencher) {
  65. divide_bench(b, 1 << 16, 1 << 12);
  66. }
  67. #[bench]
  68. fn factorial_100(b: &mut Bencher) {
  69. b.iter(|| factorial(100));
  70. }
  71. #[bench]
  72. fn fib_100(b: &mut Bencher) {
  73. b.iter(|| fib(100));
  74. }
  75. #[bench]
  76. fn fib_1000(b: &mut Bencher) {
  77. b.iter(|| fib(1000));
  78. }
  79. #[bench]
  80. fn fib_10000(b: &mut Bencher) {
  81. b.iter(|| fib(10000));
  82. }
  83. #[bench]
  84. fn fac_to_string(b: &mut Bencher) {
  85. let fac = factorial(100);
  86. b.iter(|| fac.to_string());
  87. }
  88. #[bench]
  89. fn fib_to_string(b: &mut Bencher) {
  90. let fib = fib(100);
  91. b.iter(|| fib.to_string());
  92. }
  93. fn to_str_radix_bench(b: &mut Bencher, radix: u32) {
  94. let mut rng = get_rng();
  95. let x = rng.gen_bigint(1009);
  96. b.iter(|| x.to_str_radix(radix));
  97. }
  98. #[bench]
  99. fn to_str_radix_02(b: &mut Bencher) {
  100. to_str_radix_bench(b, 2);
  101. }
  102. #[bench]
  103. fn to_str_radix_08(b: &mut Bencher) {
  104. to_str_radix_bench(b, 8);
  105. }
  106. #[bench]
  107. fn to_str_radix_10(b: &mut Bencher) {
  108. to_str_radix_bench(b, 10);
  109. }
  110. #[bench]
  111. fn to_str_radix_16(b: &mut Bencher) {
  112. to_str_radix_bench(b, 16);
  113. }
  114. #[bench]
  115. fn to_str_radix_36(b: &mut Bencher) {
  116. to_str_radix_bench(b, 36);
  117. }
  118. fn from_str_radix_bench(b: &mut Bencher, radix: u32) {
  119. use num::Num;
  120. let mut rng = get_rng();
  121. let x = rng.gen_bigint(1009);
  122. let s = x.to_str_radix(radix);
  123. assert_eq!(x, BigInt::from_str_radix(&s, radix).unwrap());
  124. b.iter(|| BigInt::from_str_radix(&s, radix));
  125. }
  126. #[bench]
  127. fn from_str_radix_02(b: &mut Bencher) {
  128. from_str_radix_bench(b, 2);
  129. }
  130. #[bench]
  131. fn from_str_radix_08(b: &mut Bencher) {
  132. from_str_radix_bench(b, 8);
  133. }
  134. #[bench]
  135. fn from_str_radix_10(b: &mut Bencher) {
  136. from_str_radix_bench(b, 10);
  137. }
  138. #[bench]
  139. fn from_str_radix_16(b: &mut Bencher) {
  140. from_str_radix_bench(b, 16);
  141. }
  142. #[bench]
  143. fn from_str_radix_36(b: &mut Bencher) {
  144. from_str_radix_bench(b, 36);
  145. }
  146. #[bench]
  147. fn shl(b: &mut Bencher) {
  148. let n = BigUint::one() << 1000;
  149. b.iter(|| {
  150. let mut m = n.clone();
  151. for i in 0..50 {
  152. m = m << i;
  153. }
  154. })
  155. }
  156. #[bench]
  157. fn shr(b: &mut Bencher) {
  158. let n = BigUint::one() << 2000;
  159. b.iter(|| {
  160. let mut m = n.clone();
  161. for i in 0..50 {
  162. m = m >> i;
  163. }
  164. })
  165. }
  166. #[bench]
  167. fn hash(b: &mut Bencher) {
  168. use std::collections::HashSet;
  169. let mut rng = get_rng();
  170. let v: Vec<BigInt> = (1000..2000).map(|bits| rng.gen_bigint(bits)).collect();
  171. b.iter(|| {
  172. let h: HashSet<&BigInt> = v.iter().collect();
  173. assert_eq!(h.len(), v.len());
  174. });
  175. }
  176. #[bench]
  177. fn pow_bench(b: &mut Bencher) {
  178. b.iter(|| {
  179. let upper = 100_usize;
  180. for i in 2..upper + 1 {
  181. for j in 2..upper + 1 {
  182. let i_big = BigUint::from_usize(i).unwrap();
  183. num::pow(i_big, j);
  184. }
  185. }
  186. });
  187. }