-
Notifications
You must be signed in to change notification settings - Fork 363
/
Copy pathfast_exponentiation.hpp
73 lines (57 loc) · 1.9 KB
/
fast_exponentiation.hpp
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
/*
Fast exponentiation
-------------------
Given the base (B) and exponent (E), this algorithm efficiently calculates the
value of B^E using the method of exponentiation by squaring, i.e, it squares
the base and halves the exponent in each step, until the exponent becomes 0.
Time complexity
---------------
O(log(E)), where E is the exponent.
Space complexity
----------------
O(log(E)), where E is the exponent.
*/
#ifndef FAST_EXPONENTIATION_HPP
#define FAST_EXPONENTIATION_HPP
#include <climits>
#include <cmath>
typedef unsigned long long ULL;
/*
digits_required
---------------
Returns the number of digits in the result of B^E.
*/
ULL digits_required(const ULL& base, const ULL& exponent) {
return floor(exponent * log10(base)) + 1;
}
/*
square
------
Returns the square of the value passed.
*/
ULL square(const ULL& value) {
return value * value;
}
/*
fast_exp
--------
Returns the value of the base raised to the exponent (B^E).
The result is modulo the third parameter passed.
NOTE: If the third parameter is not passed, it is assumed to be the maximum
value of a 64-bit integer, and the exact value of the result is returned.
In case the result cannot be contained in a 64-bit integer, the result is
automatically modulo 10^9+7.
*/
ULL fast_exp(ULL base, ULL exponent, ULL mod = ULLONG_MAX) {
if (exponent == 0)
return 1;
// if the result cannot be contained in a 64-bit integer, it will be
// modulo 10^9+7 to prevent incorrect result due to integer overflow
if (mod == ULLONG_MAX and digits_required(base, exponent) > 19)
mod = 1000000007;
if (exponent % 2) // the exponent is odd
return (fast_exp(base, exponent-1, mod) * base) % mod;
// the exponent is even
return square(fast_exp(base, exponent/2, mod)) % mod;
}
#endif // FAST_EXPONENTIATION_HPP