龙空技术网

LeetCode 力扣官方题解 | 1220. 统计元音字母序列的数目

力扣LeetCode 302

前言:

此时姐妹们对“java 统计字符个数”大约比较着重,朋友们都想要了解一些“java 统计字符个数”的相关内容。那么小编在网络上网罗了一些有关“java 统计字符个数””的相关资讯,希望同学们能喜欢,看官们快快来了解一下吧!

力扣 1220. 统计元音字母序列的数目

题目描述

给你一个整数 n,请你帮忙统计一下我们可以按下述规则形成多少个长度为 n 的字符串:

字符串中的每个字符都应当是小写元音字母('a', 'e', 'i', 'o', 'u')每个元音 'a' 后面都只能跟着 'e'每个元音 'e' 后面只能跟着 'a' 或者是 'i'每个元音 'i' 后面 不能 再跟着另一个 'i'每个元音 'o' 后面只能跟着 'i' 或者是 'u'每个元音 'u' 后面只能跟着 'a'

由于答案可能会很大,所以请你返回 模 10^9 + 7 之后的结果。

示例 1:

输入:n = 1输出:5解释:所有可能的字符串分别是:"a", "e", "i" , "o" 和 "u"。

示例 2:

输入:n = 2输出:10解释:所有可能的字符串分别是:"ae", "ea", "ei", "ia", "ie", "io", "iu", "oi", "ou" 和 "ua"。

示例 3:

输入:n = 5输出:68

提示:

1 <= n <= 2 * 10^4

解决方案

方法一:动态规划

思路

题目中给定的字符的下一个字符的规则如下:

字符串中的每个字符都应当是小写元音字母 (‘a’,‘e’,‘i’,‘o’,‘u’);每个元音 ‘a’ 后面都只能跟着 ‘e’;每个元音 ‘e’ 后面只能跟着 ‘a’ 或者是 ‘i’;每个元音 ‘i’ 后面不能再跟着另一个 ‘i’;每个元音 ‘o’ 后面只能跟着 ‘i’ 或者是 ‘u’;每个元音 ‘u’ 后面只能跟着 ‘a’;

以上等价于每个字符的前一个字符的规则如下:

元音字母 ‘a’ 前面只能跟着 ‘e’,‘i’,‘u’;元音字母 ‘e’ 前面只能跟着 ‘a’,‘i’;每个元音 ‘i’ 前面只能跟着 ‘e’,‘o’;每个元音 ‘o’ 前面只能跟着 ‘i’;每个元音 ‘u’ 后面只能跟着 ‘o’,‘i’;

我们设 dp[i][j] 代表当前长度为 i 且以字符 j 为结尾的字符串的数目,其中在此 j=0,1,2,3,4 分别代表元音字母 ‘a’,‘e’,‘i’,‘o’,‘u’,通过以上的字符规则,我们可以得到动态规划递推公式如下:

按照题目规则最终可以形成长度为 n 的字符串的数目为:

实际计算过程中只需要保留前一个状态即可推导出后一个状态,计算长度为 i 的状态只需要长度为 i−1 的中间变量即可,i−1 之前的状态全部都可以丢弃掉。计算过程中,答案需要取模 1e9+7。

代码

Java

class Solution {    public int countVowelPermutation(int n) {        long mod = 1000000007;        long[] dp = new long[5];        long[] ndp = new long[5];        for (int i = 0; i < 5; ++i) {            dp[i] = 1;        }        for (int i = 2; i <= n; ++i) {            /* a前面可以为e,u,i */            ndp[0] = (dp[1] + dp[2] + dp[4]) % mod;            /* e前面可以为a,i */            ndp[1] = (dp[0] + dp[2]) % mod;            /* i前面可以为e,o */            ndp[2] = (dp[1] + dp[3]) % mod;            /* o前面可以为i */            ndp[3] = dp[2];            /* u前面可以为i,o */            ndp[4] = (dp[2] + dp[3]) % mod;            System.arraycopy(ndp, 0, dp, 0, 5);        }        long ans = 0;        for (int i = 0; i < 5; ++i) {            ans = (ans + dp[i]) % mod;        }        return (int)ans;    }}

C++

class Solution {public:    int countVowelPermutation(int n) {        long long mod = 1e9 + 7;        vector<long long> dp(5, 1);        vector<long long> ndp(5);        for (int i = 2; i <= n; ++i) {            /* a前面可以为e,u,i */            ndp[0] = (dp[1] + dp[2] + dp[4]) % mod;            /* e前面可以为a,i */            ndp[1] = (dp[0] + dp[2]) % mod;            /* i前面可以为e,o */            ndp[2] = (dp[1] + dp[3]) % mod;            /* o前面可以为i */            ndp[3] = dp[2];            /* u前面可以为i,o */            ndp[4] = (dp[2] + dp[3]) % mod;            dp = ndp;        }        return accumulate(dp.begin(), dp.end(), 0LL) % mod;    }};

C#

public class Solution {    public int CountVowelPermutation(int n) {        long mod = 1000000007;        long[] dp = new long[5];        long[] ndp = new long[5];        for (int i = 0; i < 5; ++i) {            dp[i] = 1;        }        for (int i = 2; i <= n; ++i) {            /* a前面可以为e,u,i */            ndp[0] = (dp[1] + dp[2] + dp[4]) % mod;            /* e前面可以为a,i */            ndp[1] = (dp[0] + dp[2]) % mod;            /* i前面可以为e,o */            ndp[2] = (dp[1] + dp[3]) % mod;            /* o前面可以为i */            ndp[3] = dp[2];            /* u前面可以为i,o */            ndp[4] = (dp[2] + dp[3]) % mod;            Array.Copy(ndp, 0, dp, 0, 5);        }        long ans = 0;        for (int i = 0; i < 5; ++i) {            ans = (ans + dp[i]) % mod;        }        return (int)ans;    }}

Python3

class Solution:    def countVowelPermutation(self, n: int) -> int:        dp = (1, 1, 1, 1, 1)        for _ in range(n - 1):            dp = ((dp[1] + dp[2] + dp[4]) % 1000000007, (dp[0] + dp[2]) % 1000000007, (dp[1] + dp[3]) % 1000000007, dp[2], (dp[2] + dp[3]) % 1000000007)        return sum(dp) % 1000000007

C

typedef long long LL;int countVowelPermutation(int n){    long long mod = 1e9 + 7;    long long ans = 0;    LL * dp = (LL *)malloc(sizeof(LL) * 5);    LL * ndp = (LL *)malloc(sizeof(LL) * 5);        for (int i = 0; i < 5; ++i) {        dp[i] = 1;    }    for (int i = 2; i <= n; ++i) {        /* a前面可以为e,u,i */        ndp[0] = (dp[1] + dp[2] + dp[4]) % mod;        /* e前面可以为a,i */        ndp[1] = (dp[0] + dp[2]) % mod;        /* i前面可以为e,o */        ndp[2] = (dp[1] + dp[3]) % mod;        /* o前面可以为i */        ndp[3] = dp[2];        /* u前面可以为i,o */        ndp[4] = (dp[2] + dp[3]) % mod;        memcpy(dp, ndp, sizeof(LL) * 5);    }    for (int i = 0; i < 5; ++i) {        ans = (ans + dp[i]) % mod;    }    free(dp);    free(ndp);    return ans;}

Golang

func countVowelPermutation(n int) (ans int) {    const mod int = 1e9 + 7    dp := [5]int{1, 1, 1, 1, 1}    for i := 1; i < n; i++ {        dp = [5]int{            (dp[1] + dp[2] + dp[4]) % mod, // a 前面可以为 e,u,i            (dp[0] + dp[2]) % mod,         // e 前面可以为 a,i            (dp[1] + dp[3]) % mod,         // i 前面可以为 e,o            dp[2],                         // o 前面可以为 i            (dp[2] + dp[3]) % mod,         // u 前面可以为 i,o        }    }    for _, v := range dp {        ans = (ans + v) % mod    }    return}

JavaScript

var countVowelPermutation = function(n) {    const mod = 1000000007;    const dp = new Array(5).fill(0);    const ndp = new Array(5).fill(0);    for (let i = 0; i < 5; ++i) {        dp[i] = 1;    }    for (let i = 2; i <= n; ++i) {        /* a前面可以为e,u,i */        ndp[0] = (dp[1] + dp[2] + dp[4]) % mod;        /* e前面可以为a,i */        ndp[1] = (dp[0] + dp[2]) % mod;        /* i前面可以为e,o */        ndp[2] = (dp[1] + dp[3]) % mod;        /* o前面可以为i */        ndp[3] = dp[2];        /* u前面可以为i,o */        ndp[4] = (dp[2] + dp[3]) % mod;        dp.splice(0, 5, ...ndp);    }    let ans = 0;    for (let i = 0; i < 5; ++i) {        ans = (ans + dp[i]) % mod;    }    return ans;};

复杂度分析

时间复杂度:O(C×n)。其中 n 是给定,C 表示元音字母的数量,在本题中 C=5。空间复杂度:O(C)。我们只需要常数个空间存储不同组的数目。

方法二:矩阵快速幂

思路

已经知道上述的递推公式,可以转将其转化为矩阵乘法,设 f(n) 表示长度为 n 的字符串且以不同元音字母为结尾的组合数矩阵,构造矩阵的递推关系如下:

因此我们可以推出:

令:

因此只要我们能快速计算矩阵 M 的 n 次幂,就可以得到 f(n) 的值。如果直接求取 Mⁿ,时间复杂度是 O(n),可以定义矩阵乘法,然后用快速幂算法来加速 Mⁿ 的求取。计算过程中,答案需要取模 1e9+7。

代码

Java

class Solution {    public int countVowelPermutation(int n) {        long mod = 1_000_000_007;        long[][] factor =        {            {0, 1, 0, 0, 0},             {1, 0, 1, 0, 0},             {1, 1, 0, 1, 1},             {0, 0, 1, 0, 1},             {1, 0, 0, 0, 0}        };        long[][] res = fastPow(factor, n - 1, mod);        long ans = 0;        for (int i = 0; i < 5; ++i) {            for (int j = 0; j < 5; ++j) {                ans = (ans + res[i][j]) % mod;            }        }        return (int)ans;    }    public long[][] fastPow(long[][] matrix, int n, long mod) {        int m = matrix.length;        long[][] res = new long[m][m];        long[][] curr = matrix;        for (int i = 0; i < m; ++i) {            res[i][i] = 1;        }        for (int i = n; i != 0; i >>= 1) {            if ((i % 2) == 1) {                res = multiply(curr, res, mod);            }            curr = multiply(curr, curr, mod);        }        return res;    }    public long[][] multiply(long[][] matrixA, long[][] matrixB, long mod) {        int m = matrixA.length;        int n = matrixB[0].length;        long[][] res = new long[m][n];        for (int i = 0; i < m; ++i) {            for (int j = 0; j < n; ++j) {                res[i][j] = 0;                for (int k = 0; k < matrixA[i].length; ++k) {                    res[i][j] = (res[i][j] + matrixA[i][k] * matrixB[k][j]) % mod;                }            }        }        return res;    }}

C++

using LL = long long;using Mat = vector<vector<LL>>;class Solution {public:         Mat multiply(const Mat & matrixA, const Mat & matrixB, LL mod) {        int m = matrixA.size();        int n = matrixB[0].size();        Mat res(m, vector<LL>(n, 0));        for (int i = 0; i < m; ++i) {            for (int j = 0; j < n; ++j) {                for (int k = 0; k < matrixA[i].size(); ++k) {                    res[i][j] = (res[i][j] + matrixA[i][k] * matrixB[k][j]) % mod;                }            }        }        return res;    }         Mat fastPow(const Mat & matrix, LL n, LL mod) {        int m = matrix.size();        Mat res(m, vector<LL>(m, 0));        Mat curr = matrix;        for (int i = 0; i < m; ++i) {            res[i][i] = 1;        }        for (int i = n; i != 0; i >>= 1) {            if (i & 1) {                res = multiply(curr, res, mod);            }            curr = multiply(curr, curr, mod);        }        return res;    }    int countVowelPermutation(int n) {        LL mod = 1e9 + 7;        Mat factor =        {            {0, 1, 0, 0, 0},             {1, 0, 1, 0, 0},             {1, 1, 0, 1, 1},             {0, 0, 1, 0, 1},             {1, 0, 0, 0, 0}        };        Mat res = fastPow(factor, n - 1, mod);        long long ans = 0;        for (int i = 0; i < 5; ++i) {            ans = (ans + accumulate(res[i].begin(), res[i].end(), 0LL)) % mod;        }        return  ans;    }};

C#

public class Solution {    public int CountVowelPermutation(int n) {        long mod = 1_000_000_007;        long[][] factor =        {            new long[]{0, 1, 0, 0, 0},             new long[]{1, 0, 1, 0, 0},             new long[]{1, 1, 0, 1, 1},             new long[]{0, 0, 1, 0, 1},             new long[]{1, 0, 0, 0, 0}        };        long[][] res = FastPow(factor, n - 1, mod);        long ans = 0;        for (int i = 0; i < 5; ++i) {            for (int j = 0; j < 5; ++j) {                ans = (ans + res[i][j]) % mod;            }        }        return (int)ans;    }    public long[][] FastPow(long[][] matrix, int n, long mod) {        int m = matrix.Length;        long[][] res = new long[m][];        for (int i = 0; i < m; ++i) {            res[i] = new long[m];        }        long[][] curr = matrix;        for(int i = 0; i < m; ++i) {            res[i][i] = 1;        }        for (int i = n; i != 0; i >>= 1) {            if ((i % 2) == 1) {                res = Multiply(curr, res, mod);            }            curr = Multiply(curr, curr, mod);        }        return res;    }    public long[][] Multiply(long[][] matrixA, long[][] matrixB, long mod) {        int m = matrixA.Length;        int n = matrixB[0].Length;        long[][] res = new long[m][];        for (int i = 0; i < m; ++i) {            res[i] = new long[n];        }        for (int i = 0; i < m; ++i) {            for (int j = 0; j < n; ++j) {                res[i][j] = 0;                for (int k = 0; k < matrixA[i].Length; ++k) {                    res[i][j] = (res[i][j] + matrixA[i][k] * matrixB[k][j]) % mod;                }            }        }        return res;    }}

Python3

import numpy as npclass Solution:    def countVowelPermutation(self, n: int) -> int:        factor = np.mat([(0, 1, 0, 0, 0), (1, 0, 1, 0, 0), (1, 1, 0, 1, 1), (0, 0, 1, 0, 1), (1, 0, 0, 0, 0)], np.dtype('O'))        res = np.mat([(1, 1, 1, 1, 1)], np.dtype('O'))        n -= 1        while n > 0:            if n % 2 == 1:                res = res * factor % 1000000007            factor = factor * factor % 1000000007            n = n // 2        return res.sum() % 1000000007

C

typedef long long LL;typedef LL * Mat;#define IDX(x, y, col) ((x) * (col) + (y))Mat multiply(const Mat matrixA, int matrixARowSize, int matrixAColSize, const Mat matrixB, int matrixBRowSize, int matrixBColSize, LL mod) {    Mat res = (LL *)malloc(sizeof(LL) * matrixARowSize * matrixBColSize);    memset(res, 0, sizeof(LL) * matrixARowSize * matrixBColSize);    for (int i = 0; i < matrixARowSize; ++i) {        for (int j = 0; j < matrixBColSize; ++j) {            for (int k = 0; k < matrixAColSize; ++k) {                res[IDX(i, j, matrixAColSize)] = (res[IDX(i, j, matrixAColSize)] + \                                                  matrixA[IDX(i, k, matrixAColSize)] * \                                                  matrixB[IDX(k, j, matrixBColSize)]) % mod;            }        }    }    return res;}Mat fastPow(const Mat matrix, int matrixRowSize, LL n, LL mod) {    Mat res = (LL *)malloc(sizeof(LL) * matrixRowSize * matrixRowSize);    Mat curr = (LL *)malloc(sizeof(LL) * matrixRowSize * matrixRowSize);    memset(res, 0, sizeof(LL) * matrixRowSize * matrixRowSize);    memcpy(curr, matrix, sizeof(LL) * matrixRowSize * matrixRowSize);    for (int i = 0; i < matrixRowSize; ++i) {        res[IDX(i, i, matrixRowSize)] = 1;    }    for (int i = n; i != 0; i >>= 1) {        if (i & 1) {            Mat nextRes = multiply(curr, matrixRowSize, matrixRowSize, res, matrixRowSize, matrixRowSize, mod);            free(res);            res = nextRes;        }        Mat nextCurr = multiply(curr, matrixRowSize, matrixRowSize, curr, matrixRowSize, matrixRowSize, mod);        free(curr);        curr = nextCurr;    }    free(curr);    return res;}int countVowelPermutation(int n){    LL mod = 1e9 + 7;    LL factor[25] = { \                      0, 1, 0, 0, 0, \                      1, 0, 1, 0, 0, \                      1, 1, 0, 1, 1, \                      0, 0, 1, 0, 1, \                      1, 0, 0, 0, 0  \                    };    Mat res = fastPow(factor, 5, n - 1, mod);    LL ans = 0;    for (int i = 0; i < 25; ++i) {        ans = (ans + res[i]) % mod;    }    free(res);    return ans;}

Golang

const mod int = 1e9 + 7type matrix [5][5]intfunc (a matrix) mul(b matrix) matrix {    c := matrix{}    for i, row := range a {        for j := range b[0] {            for k, v := range row {                c[i][j] = (c[i][j] + v*b[k][j]) % mod            }        }    }    return c}func (a matrix) pow(n int) matrix {    res := matrix{}    for i := range res {        res[i][i] = 1    }    for ; n > 0; n >>= 1 {        if n&1 > 0 {            res = res.mul(a)        }        a = a.mul(a)    }    return res}func countVowelPermutation(n int) (ans int) {    m := matrix{        {0, 1, 0, 0, 0},        {1, 0, 1, 0, 0},        {1, 1, 0, 1, 1},        {0, 0, 1, 0, 1},        {1, 0, 0, 0, 0},    }    res := m.pow(n - 1)    for _, row := range res {        for _, v := range row {            ans = (ans + v) % mod        }    }    return}

复杂度分析

时间复杂度:初始化为 O(C³logn)。其中 n 是给定的参数,C 表示元音字母的数量,在本题中 C=5,每次矩阵相乘的时间复杂度为 O(C³),最多需要logn 次矩阵相乘。空间复杂度:O(C²)。需要保空间来存储矩阵乘法的结果。

BY /

本文作者:力扣

声明:本文归“力扣”版权所有,如需转载请联系。

标签: #java 统计字符个数 #java获取字符串中字符个数