zoukankan      html  css  js  c++  java
  • [BZOJ]4162: shlw loves matrix II

    Time Limit: 30 Sec  Memory Limit: 128 MB

    Description

      给定矩阵 M,请计算 M^n,并将其中每一个元素对 1000000007 取模输出。

    Input

      第 1 行包含两个整数 n,k,其中 n 使用二进制表示,可能含有前导零;
      余下 k 行描述了一个 k * k 的矩阵 M。

    Output

      输出题目描述中要求的矩阵,格式同输入。

    Sample Input

      010 3
      5 9 5
      5 4 0
      8 8 8

    Sample Output

      110 121 65
      45 61 25
      144 168 104

    HINT

      对于 100% 数据,满足 n <= 2^10000;k <= 50; 0 <= Mij < 10^9 +7

    Solution

      矩阵乘法特征多项式优化,矩阵M的特征多项式是$f(lambda)=|M-lambda I|$,随便带入k+1个$lambda$,高斯消元求出行列式的值,然后插值就能求出M的特征多项式(这里的总复杂度为$O(k^4)$,主要在计算行列式上面)。根据Cayley-Hamilton定理,$f(M)=0$,也就是说对同一个式子减去若干个$f(M)$后值不变,我们即可用M的k-1次多项式表示一个M的次幂,两个多项式相乘后对$f(M)$取模,然后就可以快速幂了,多项式取模和乘法用$O(k^2)$暴力就够了,总复杂度$O(k^4+k^2logn)$。

    Code

    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    using namespace std;
    #define ll long long
    inline int read()
    {
        int x;char c;
        while((c=getchar())<'0'||c>'9');
        for(x=c-'0';(c=getchar())>='0'&&c<='9';)x=x*10+c-'0';
        return x;
    }
    #define ML 10000
    #define MK 50
    #define MN 100
    #define MOD 1000000007
    char s[ML+5];
    int k,c[MK+5][MK+5],x[MK+5],ans[MK+5][MK+5];
    inline int mo1(int x){return x<MOD?x:x-MOD;}
    inline int mo2(int x){return x<0?x+MOD:x;}
    int inv(int x)
    {
        int r=1,y=MOD-2;
        for(;y;y>>=1,x=1LL*x*x%MOD)if(y&1)r=1LL*r*x%MOD;
        return r;
    }
    struct poly
    {
        int n,a[MN+5];
        poly(int n=0,int a0=0,int a1=0):n(n){memset(a,0,sizeof(a));a[0]=a0;a[1]=a1;}
        friend poly operator+(poly a,const poly&b)
        {
            a.n=max(a.n,b.n);
            for(int i=0;i<=a.n;++i)a.a[i]=mo1(a.a[i]+b.a[i]);
            return a;
        }
        friend poly operator*(const poly&a,const poly&b)
        {
            poly c(a.n+b.n);int i,j;ll x;
            for(i=0;i<=c.n;c.a[i++]=x%MOD)for(x=j=0;j<=i&&j<=a.n;++j)
                x+=1LL*a.a[j]*b.a[i-j],x>8e18?x%=MOD:0;
            return c;
        }
        friend poly operator*(poly a,int b)
        {
            for(int i=0;i<=a.n;++i)a.a[i]=1LL*a.a[i]*b%MOD;
            return a;
        }
        friend poly operator/(poly a,const poly&b)
        {
            poly c(a.n-b.n);int i,j;
            for(i=a.n;i>=b.n;--i)
            {
                c.a[i-b.n]=1LL*a.a[i]*inv(b.a[b.n])%MOD;
                for(j=1;j<=b.n;++j)a.a[i-j]=mo2((a.a[i-j]-1LL*b.a[b.n-j]*c.a[i-b.n])%MOD);
            }
            return c;
        }
        friend poly operator%(poly a,const poly&b)
        {
            int i,j,x;
            for(i=a.n;i>=b.n;--i)
            {
                x=1LL*a.a[i]*inv(b.a[b.n])%MOD;
                for(j=0;j<=b.n;++j)a.a[i-j]=mo2((a.a[i-j]-1LL*b.a[b.n-j]*x)%MOD);
            }
            a.n=b.n-1;
            return a;
        }
    }f,a,p(0,1);
    struct mat
    {
        int z[MK+5][MK+5];
        mat(){memset(z,0,sizeof(z));}
        friend mat operator*(const mat&a,const mat&b)
        {
            mat c;int i,j,k;ll x;
            for(i=0;i<=MK;++i)for(j=0;j<=MK;c.z[i][j++]=x%MOD)
                for(x=k=0;k<=MK;++k)x+=1LL*a.z[i][k]*b.z[k][j],x>8e18?x%=MOD:0;
            return c;
        }
    }m,n;
    int cal(int x)
    {
        int i,j,l,ans=1;
        for(i=1;i<=k;++i)for(j=1;j<=k;++j)c[i][j]=m.z[i][j];
        for(i=1;i<=k;++i)c[i][i]=mo2(c[i][i]-x);
        for(i=1;i<=k;++i)
        {
            for(j=i;j<=k;++j)if(c[j][i])break;
            if(j>k)return 0;
            if(j>i)for(ans=MOD-ans,l=i;l<=k;++l)swap(c[i][l],c[j][l]);
            ans=1LL*ans*c[i][i]%MOD;
            for(j=i;++j<=k;)
            {
                x=1LL*c[j][i]*inv(c[i][i])%MOD;
                for(l=i;l<=k;++l)c[j][l]=mo2((c[j][l]-1LL*c[i][l]*x)%MOD);
            }
        }
        return ans;
    }
    int main()
    {
        int i,j,l,v;
        scanf("%s",s+1);k=read();
        for(i=1;i<=k;++i)for(j=1;j<=k;++j)m.z[i][j]=read();
        for(i=0;i<=k;++i)x[i]=cal(i);
        for(i=0;i<=k;++i)p=p*poly(1,mo2(-i),1);
        for(i=0;i<=k;++i)
        {
            for(v=x[i],j=0;j<=k;++j)if(i!=j)v=1LL*v*inv(mo2(i-j))%MOD;
            f=f+p/poly(1,mo2(-i),1)*v;
        }
        a=poly(0,1);p=poly(1,0,1);
        for(i=strlen(s+1);i;--i,p=p*p%f)if(s[i]>'0')a=a*p%f;
        for(i=0;i<=k;++i)n.z[i][i]=1;
        for(l=0;l<k;++l,n=n*m)for(i=1;i<=k;++i)for(j=1;j<=k;++j)
            ans[i][j]=(ans[i][j]+1LL*a.a[l]*n.z[i][j])%MOD;
        for(i=1;i<=k;printf("%d
    ",ans[i++][j]))for(j=1;j<k;++j)printf("%d ",ans[i][j]);
    }
  • 相关阅读:
    seeting菜单界面形成--优化
    setting菜单界面的形成--未优化
    (转)最强Android模拟器genymotion的安装与配置
    (转)Android SlidingTabLayout定制分割线和指示条颜色
    【前端】CSS入门笔记
    【前端】XHTML入门笔记
    【Java】Java学习笔记
    【英语】20141022 生词
    【学习】Git和Github菜鸟入门
    【英语】20141015 生词
  • 原文地址:https://www.cnblogs.com/ditoly/p/BZOJ4162.html
Copyright © 2011-2022 走看看