洛谷P1939 矩阵快速幂模板

矩阵快速幂解线性递推式:

a n = a n − 1 + a n − 3 a_{n}=a_{n-1}+a_{n-3} an=an1+an3

Solution:

一般构造矩阵的时候都构造一个连续的矩阵,且这个矩阵元素下标都 + 1 +1 +1后两个矩阵恰好覆盖了递推式所涉及的所有项,设递推式跨越的下标长度为 l e n len len,这个长度一般是 l e n − 1 len-1 len1,譬如当前构造矩阵
[ a n , a n − 1 , a n − 2 ] [a_{n},a_{n-1},a_{n-2}] [an,an1,an2]
我们需要构造出一个矩阵,使得这个上式矩阵乘构造矩阵后得到
[ a n + 1 , a n , a n − 1 ] [a_{n+1},a_{n},a_{n-1}] [an+1,an,an1]

矩阵乘法:

对于两个矩阵 a , b a,b a,b相乘,结果矩阵的 i i i j j j列为
∑ k = 1 n a i , k b k , j \sum_{k=1}^{n}a_{i,k}b_{k,j} k=1nai,kbk,j
a a a i i i行与 b b b j j j列的所有元素对应相乘相加

单位矩阵:

是主对角线元素为 1 1 1,其余元素为 0 0 0的矩阵,即满足下式的矩阵
a i , j = { 0 i ≠ j 1 i = j a_{i,j}=\begin{cases} 0 & i\neq j\\ 1 & i=j \end{cases} ai,j={01i=ji=j

根据矩阵乘法的规则构造矩阵为
[ 1 1 0 0 0 1 1 0 0 ] \left[ \begin{matrix} 1 & 1 & 0\\ 0 & 0 & 1\\ 1 & 0 & 0 \end{matrix} \right] 101100010
对此矩阵快速幂即可

需要注意的是,单位矩阵是主对角线元素为 1 1 1的矩阵,其他元素为 0 0 0;而不是都是 1 1 1的矩阵

// #include
#include
#include
#include
#include
#include
#include
#include
#include
using namespace std;

using ll=long long;
const int N=1e6+5,inf=0x3fffffff;
const long long INF=0x3fffffffffffffff,mod=1e9+7;

struct matrix
{
    ll a[3][3];
    void initial(){memset(a,0,sizeof(a));}
    void build()
    {
        initial();
        for(int i=0;i<3;i++) a[i][i]=1;
    }
    ll* operator[](int i) const {
        return (ll*)&a[i][0];
    }
    matrix operator*(const matrix& x) const
    {
        matrix ret; ret.initial();
        for(int i=0;i<3;i++)
            for(int j=0;j<3;j++)
                for(int k=0;k<3;k++) ret[i][j]=(ret[i][j]+a[i][k]*x[k][j]%mod)%mod;
        return ret;
    }
};

matrix qpow(matrix a,ll b)
{
    matrix ret,base=a; ret.build();
    while(b)
    {
        if(b&1) ret=ret*base;
        base=base*base;
        b>>=1;
    }
    return ret;
}

int main()
{
    matrix tmp={{
        {1,1,0},
        {0,0,1},
        {1,0,0}
    }},ans={{
        {1,1,1},
        {0,0,0},
        {0,0,0}
    }};
    int t; cin>>t;
    while(t--)
    {
        ll n; cin>>n;
        if(n<=3) cout<<1<<'\n';
        else cout<<(ans*qpow(tmp,n-3)).a[0][0]<<'\n';
    }
    return 0;  
}

你可能感兴趣的:(算法)