diff --git a/guava/src/com/google/common/math/BigIntegerMath.java b/guava/src/com/google/common/math/BigIntegerMath.java index ada0f0f57c80..30f737b7850a 100644 --- a/guava/src/com/google/common/math/BigIntegerMath.java +++ b/guava/src/com/google/common/math/BigIntegerMath.java @@ -516,6 +516,30 @@ public static BigInteger binomial(int n, int k) { static boolean fitsInLong(BigInteger x) { return x.bitLength() <= Long.SIZE - 1; } - + /** + * Returns multifactorial of n with step size BigInteger k. + * This is used for large values of n. + * + * @param n the number to compute the multifactorial of. Must be non-negative. + * @param k the step size. Must be positive. + * @return the multifactorial of n with step size k. If none-zero n is less than k then return n, else return one. + * @throws IllegalArgumentException if n is negative or if k is less than 1. + */ + public static BigInteger multiFactorial(int n, int k) { + if (n < 0) { + throw new IllegalArgumentException("n cannot be negative!"); + } + if (k < 1) { + throw new IllegalArgumentException("k must be positive!"); + } + if (n <= k) { + return n == 0 ? BigInteger.ONE : BigInteger.valueOf(n); + } + BigInteger result = BigInteger.valueOf(n); + for (int i = n - k; i > 1; i -= k) { + result = result.multiply(BigInteger.valueOf(i)); + } + return result; + } private BigIntegerMath() {} } diff --git a/guava/src/com/google/common/math/IntMath.java b/guava/src/com/google/common/math/IntMath.java index 74b6de68a681..0f8389b38569 100644 --- a/guava/src/com/google/common/math/IntMath.java +++ b/guava/src/com/google/common/math/IntMath.java @@ -717,6 +717,30 @@ public static int mean(int x, int y) { public static boolean isPrime(int n) { return LongMath.isPrime(n); } + /** + * Returns multifactorial of Int n with step size k. + * + * @param n the number to compute. Must be non-negative. + * @param k the step size for the multifactorial. Must be positive. + * @return the multifactorial of n with step size k. If none-zero n is less than k then return n, else return 1. + * @throws IllegalArgumentException if n is negative or if k is less than 1. + */ + public static int multiFactorial(int n, int k) { + if (n < 0) { + throw new IllegalArgumentException("n cannot be negative!"); + } + if (k < 1) { + throw new IllegalArgumentException("k must be positive!"); + } + if (n <= k) { + return n == 0 ? 1 : n; + } + int result = n; + for (int i = n - k; i > 1; i -= k) { + result *= i; + } + return result; + } private IntMath() {} } diff --git a/guava/src/com/google/common/math/LongMath.java b/guava/src/com/google/common/math/LongMath.java index 16e0248d99e4..790526969c58 100644 --- a/guava/src/com/google/common/math/LongMath.java +++ b/guava/src/com/google/common/math/LongMath.java @@ -1345,6 +1345,29 @@ public static double roundToDouble(long x, RoundingMode mode) { } throw new AssertionError("impossible"); } - + /** + * Returns multifactorial of int n with step size k. + * + * @param n the number to compute. Must be non-negative. + * @param k the step size must be positive. + * @return the long type multifactorial of n with step size k. If none-zero n is less than k then return (long) n, else return 1L. + * @throws IllegalArgumentException if n is negative or if k is less than 1. + */ + public static long multiFactorial(int n, int k) { + if (n < 0) { + throw new IllegalArgumentException("n cannot be negative!"); + } + if (k < 1) { + throw new IllegalArgumentException("k must be positive!"); + } + if (n <= k) { + return n == 0 ? 1L : (long)n; + } + long result = n; + for (long i = n - k; i > 1; i -= k) { + result *= i; + } + return result; + } private LongMath() {} }