HDU 4602 Partition

矩阵快速幂

#include<cstdio>
#include<cstring>
#include<cmath>
#include<map>
#include<algorithm>
using namespace std;

const long long MOD=1000000000+7;
long long n,k,RR,CC;
map<long long,long long>ans;
struct Matrix
{
    long long A[5][5];
    Matrix operator*(Matrix b);
};

Matrix Matrix::operator*(Matrix b)
{
    Matrix c;
    long long i,j,k;
    memset(c.A,0,sizeof(c.A));
    for(i=1;i<=RR;i++)
        for(j=1;j<=CC;j++)
            for(k=1;k<=3;k++)
                c.A[i][j]=(c.A[i][j]+(A[i][k]*b.A[k][j]+MOD)%MOD)%MOD;
    return c;
}

int main()
{
    ans.clear();
    ans[1]=1;ans[2]=2;ans[3]=5,ans[4]=12;
    int T;
    scanf("%d",&T);
    while(T--)
    {
        scanf("%lld%lld",&n,&k);
        if(k>n) {printf("0\n");continue;}
        long long u=n-k+1;
        if(ans[u]!=0) printf("%lld\n",ans[u]);
        else
        {
            long long i,j;
            long long b=u-4;
            Matrix JZ;
            JZ.A[1][1]=4;JZ.A[1][2]=0;JZ.A[1][3]=0;
            JZ.A[2][1]=-4;JZ.A[2][2]=4;JZ.A[2][3]=1;
            JZ.A[3][1]=0;JZ.A[3][2]=-4;JZ.A[3][3]=0;
            Matrix C;
            for(i=1;i<=3;i++)
                for(j=1;j<=3;j++)
                {
                    if(i==j) C.A[i][j]=1;
                    else C.A[i][j]=0;
                }
            RR=3;CC=3;
            while(b>0)
            {
                if(b%2==1)C=C*JZ,b--;
                else JZ=JZ*JZ,b=b/2;
            }

            RR=1;CC=3;
            Matrix CS;
            CS.A[1][1]=ans[4];
            CS.A[1][2]=ans[3];
            CS.A[1][3]=ans[2];
            CS=CS*C;
            ans[u]=CS.A[1][1];
            printf("%lld\n",ans[u]);
        }
    }
    return 0;
}

 

posted @ 2015-07-22 16:11  Fighting_Heart  阅读(154)  评论(0编辑  收藏  举报