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())<''||c>'');
for(x=c-'';(c=getchar())>=''&&c<='';)x=x*+c-'';
return x;
}
#define ML 10000
#define MK 50
#define MN 100
#define MOD 1000000007
char s[ML+];
int k,c[MK+][MK+],x[MK+],ans[MK+][MK+];
inline int mo1(int x){return x<MOD?x:x-MOD;}
inline int mo2(int x){return x<?x+MOD:x;}
int inv(int x)
{
int r=,y=MOD-;
for(;y;y>>=,x=1LL*x*x%MOD)if(y&)r=1LL*r*x%MOD;
return r;
}
struct poly
{
int n,a[MN+];
poly(int n=,int a0=,int a1=):n(n){memset(a,,sizeof(a));a[]=a0;a[]=a1;}
friend poly operator+(poly a,const poly&b)
{
a.n=max(a.n,b.n);
for(int i=;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=;i<=c.n;c.a[i++]=x%MOD)for(x=j=;j<=i&&j<=a.n;++j)
x+=1LL*a.a[j]*b.a[i-j],x>8e18?x%=MOD:;
return c;
}
friend poly operator*(poly a,int b)
{
for(int i=;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=;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=;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-;
return a;
}
}f,a,p(,);
struct mat
{
int z[MK+][MK+];
mat(){memset(z,,sizeof(z));}
friend mat operator*(const mat&a,const mat&b)
{
mat c;int i,j,k;ll x;
for(i=;i<=MK;++i)for(j=;j<=MK;c.z[i][j++]=x%MOD)
for(x=k=;k<=MK;++k)x+=1LL*a.z[i][k]*b.z[k][j],x>8e18?x%=MOD:;
return c;
}
}m,n;
int cal(int x)
{
int i,j,l,ans=;
for(i=;i<=k;++i)for(j=;j<=k;++j)c[i][j]=m.z[i][j];
for(i=;i<=k;++i)c[i][i]=mo2(c[i][i]-x);
for(i=;i<=k;++i)
{
for(j=i;j<=k;++j)if(c[j][i])break;
if(j>k)return ;
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+);k=read();
for(i=;i<=k;++i)for(j=;j<=k;++j)m.z[i][j]=read();
for(i=;i<=k;++i)x[i]=cal(i);
for(i=;i<=k;++i)p=p*poly(,mo2(-i),);
for(i=;i<=k;++i)
{
for(v=x[i],j=;j<=k;++j)if(i!=j)v=1LL*v*inv(mo2(i-j))%MOD;
f=f+p/poly(,mo2(-i),)*v;
}
a=poly(,);p=poly(,,);
for(i=strlen(s+);i;--i,p=p*p%f)if(s[i]>'')a=a*p%f;
for(i=;i<=k;++i)n.z[i][i]=;
for(l=;l<k;++l,n=n*m)for(i=;i<=k;++i)for(j=;j<=k;++j)
ans[i][j]=(ans[i][j]+1LL*a.a[l]*n.z[i][j])%MOD;
for(i=;i<=k;printf("%d\n",ans[i++][j]))for(j=;j<k;++j)printf("%d ",ans[i][j]);
}

[BZOJ]4162: shlw loves matrix II的更多相关文章

  1. [bzoj4162]shlw loves matrix II

    来自FallDream的博客,未经允许,请勿转载,谢谢 给定矩阵k*k的矩阵M,请计算 M^n,并将其中每一个元素对 1000000007 取模输出. k<=50 n<=2^10000 考 ...

  2. BZOJ4162:shlw loves matrix II

    传送门 利用Cayley-Hamilton定理,用插值法求出特征多项式 \(P(x)\) 然后 \(M^n\equiv M^n(mod~P(x))(mod~P(x))\) 然后就多项式快速幂+取模 最 ...

  3. bzoj 3462: DZY Loves Math II

    3462: DZY Loves Math II Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 211  Solved: 103[Submit][Sta ...

  4. bzoj 4161 Shlw loves matrixI——常系数线性齐次递推

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4161 还是不能理解矩阵…… 关于不用矩阵理解的方法:https://blog.csdn.ne ...

  5. BZOJ 3569 DZY Loves Chinese II

    Description 神校XJ之学霸兮,Dzy皇考曰JC. 摄提贞于孟陬兮,惟庚寅Dzy以降. 纷Dzy既有此内美兮,又重之以修能. 遂降临于OI界,欲以神力而凌♂辱众生. 今Dzy有一魞歄图,其上 ...

  6. BZOJ 3569: DZY Loves Chinese II [高斯消元XOR 神题]

    http://www.lydsy.com/JudgeOnline/problem.php?id=3569 题意:多次询问一个无向连通图当图中某k条边消失时这个图是否联通 强制在线 太神啦啦啦啦啦啦啦啦 ...

  7. [bzoj4161]Shlw loves matrix I

    来自FallDream的博客,未经允许,请勿转载,谢谢. 给定数列 {hn}前k项,其后每一项满足 hn = a1*h(n-1) + a2*h(n-2) + ... + ak*h(n-k) 其中 a1 ...

  8. bzoj 4161: Shlw loves matrixI

    Description 给定数列 {hn}前k项,其后每一项满足 hn = a1h(n-1) + a2h(n-2) + ... + ak*h(n-k) 其中 a1,a2...ak 为给定数列.请计算 ...

  9. BZOJ 3569 DZY Loves Chinese II 树上差分+线性基

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=3569 Description 神校XJ之学霸兮,Dzy皇考曰JC. 摄提贞于孟陬兮,惟庚寅 ...

随机推荐

  1. 2017-2018-1 20155306 mypwd的实现

    2017-2018-1 20155306 mypwd的实现 一.pwd的使用 功能: Linux中用 pwd 命令来查看"当前工作目录"的完整路径. 命令格式:pwd [选项] 命 ...

  2. 关于DLL的创建与使用简单描述(C++、C#)

    前言 前一段时间在学关于DLL的创建与调用,结果发现网络上一大堆别人分享的经验都有点问题.现在整理分享一下自己的方法. 工具 Microsoft Visual Studio 2017 depends ...

  3. NetFPGA-1G-CML Demo --- openflow_switch

    环境 ubuntu 14.04 vivado 15.2 ise 14.6 更多基础配置:http://www.cnblogs.com/wpqwpq/p/6771568.html 运行步骤 step1: ...

  4. LeetCode & Q88-Merge Sorted Array-Easy

    Array Two Pointers Description: Given two sorted integer arrays nums1 and nums2, merge nums2 into nu ...

  5. 鼠标滑过切换div显示(鼠标事件)

    <html> <head> <meta http-equiv="Content-Type" content="text/html; char ...

  6. linux 进程间通信的3种高级方式及优缺点

    由于不同的进程运行在各自不同的内存空间中.一方对于变量的修改另一方是无法感知的.因此.进程之间的信息传递不可能通过变量或其它数据结构直接进行,只能通进程间通信来完成. 根据进程通信时信息量大小的不同, ...

  7. Python内置函数(36)——reversed

    英文文档: reversed(seq) Return a reverse iterator. seq must be an object which has a __reversed__() meth ...

  8. vue-cli webpack3扩展多模块打包

    场景 在实际的项目开发中会出现这样的场景,项目中需要多个模块(单页或者多页应用)配合使用的情况,而vue-cli默认只提供了单入口打包,所以就想到对vue-cli进行扩展 实现 首先得知道webpac ...

  9. Spring中报"Could not resolve placeholder"的解决方案

    除去properites文件路径错误.拼写错误外,出现"Could not resolve placeholder"很有可能是使用了多个PropertyPlaceholderCon ...

  10. mysql Access denied for user root@localhost错误解决方法

    select * from user \G use mysql select * from user limit 1 \G update user set Host='%' where `User`= ...