|
| 1 | +// Time: O(logn) |
| 2 | +// Space: O(1) |
| 3 | + |
| 4 | +class Solution { |
| 5 | +public: |
| 6 | + int countVowelPermutation(int n) { |
| 7 | + vector<vector<int>> T = {{0, 1, 1, 0, 1}, |
| 8 | + {1, 0, 1, 0, 0}, |
| 9 | + {0, 1, 0, 1, 0}, |
| 10 | + {0, 0, 1, 0, 0}, |
| 11 | + {0, 0, 1, 1, 0}}; |
| 12 | + |
| 13 | + const auto& result = matrixExpo(T, n - 1); |
| 14 | + return accumulate(result.cbegin(), result.cend(), 0, |
| 15 | + [&](int total, const vector<int>& row) { |
| 16 | + return (total + |
| 17 | + accumulate(row.cbegin(), row.cend(), 0, |
| 18 | + [&](int sum, int val) { |
| 19 | + return (sum + val) % MOD; |
| 20 | + })) % MOD; |
| 21 | + }); |
| 22 | + } |
| 23 | + |
| 24 | +private: |
| 25 | + vector<vector<int>> matrixExpo(const vector<vector<int>>& A, int pow) { |
| 26 | + vector<vector<int>> result(A.size(), vector<int>(A.size())); |
| 27 | + vector<vector<int>> A_exp(A); |
| 28 | + for (int i = 0; i < A.size(); ++i) { |
| 29 | + result[i][i] = 1; |
| 30 | + } |
| 31 | + while (pow) { |
| 32 | + if (pow % 2 == 1) { |
| 33 | + result = matrixMult(result, A_exp); |
| 34 | + } |
| 35 | + A_exp = matrixMult(A_exp, A_exp); |
| 36 | + pow /= 2; |
| 37 | + } |
| 38 | + return result; |
| 39 | + } |
| 40 | + |
| 41 | + vector<vector<int>> matrixMult(const vector<vector<int>>& A, const vector<vector<int>>& B) { |
| 42 | + vector<vector<int>> result(A.size(), vector<int>(A.size())); |
| 43 | + for (int i = 0; i < A.size(); ++i) { |
| 44 | + for (int j = 0; j < B[0].size(); ++j) { |
| 45 | + int64_t entry = 0; |
| 46 | + for (int k = 0; k < B.size(); ++k) { |
| 47 | + entry = (static_cast<int64_t>(A[i][k]) * B[k][j] % MOD + entry) % MOD; |
| 48 | + } |
| 49 | + result[i][j] = static_cast<int>(entry); |
| 50 | + } |
| 51 | + } |
| 52 | + return result; |
| 53 | + } |
| 54 | + const int MOD = 1e9 + 7; |
| 55 | +}; |
| 56 | + |
| 57 | +// Time: O(n) |
| 58 | +// Space: O(1) |
| 59 | +class Solution2 { |
| 60 | +public: |
| 61 | + int countVowelPermutation(int n) { |
| 62 | + int a = 1, e = 1, i = 1, o = 1, u = 1; |
| 63 | + for (int x = 1; x < n; ++x) { |
| 64 | + tie(a, e, i, o, u) = make_tuple(((e + i) % MOD + u) % MOD, |
| 65 | + (a + i) % MOD, |
| 66 | + (e + o) % MOD, |
| 67 | + i, |
| 68 | + (i + o) % MOD); |
| 69 | + } |
| 70 | + return (((a + e) % MOD + (i + o) % MOD) % MOD + u) % MOD; |
| 71 | + } |
| 72 | + |
| 73 | +private: |
| 74 | + const int MOD = 1e9 + 7; |
| 75 | +}; |
0 commit comments