BZOJ 2973 石头游戏 矩乘加速递推

FFFFFFF,看了一上午才看懂,又调了一中午。。。。。我终于明白为何自己如此菜了qwq


 

这个题加速的思路是:因为每个序列的长度小于6,他们的lcm是60,所以六十次以后就会回到原来的序列。

加速的就是这一个个重复的60次

我们把60个转移矩阵乘起来(结合律),设为d,然后有x=t/60就是有多少个d,算出d的x次方(快速幂)

然后不足60次的一个个乘起来就好了

至于如何建转移矩阵。。。模拟一下吧(我搞了一上午qwq):

e[k]是第k次的转移矩阵,取石子从0里面取(因此e[k][0][0]都是1)

对于这个矩阵,可以理解为 e[第k次][从哪个状态来][到哪个状态去]

因为矩阵乘不就是ret[i][j]+=a[i][k]*b[k][j],其中b就是转移矩阵(再不理解可以吧i那一维给去了)

PS:sizeof时注意是指针的大小还是数组的大小。。。因为这个调了一中午。。。。

#include<cstdio>
#include<iostream>
#include<cstring>
#define ll long long
#define R register int
using namespace std;
inline ll g() {
    register ll ret=0,fix=1; register char ch; while(!isdigit(ch=getchar())) fix=ch=='-'?-1:fix;
    do ret=ret*10+(ch^48); while(isdigit(ch=getchar())); return ret*fix;
}
int n,m,t,p,q;
int a[20][20],cnt[20][20];
ll f[70],d[70][70],e[70][70][70];
ll ans;
char b[20][20],s[100],ch;
inline int pos(int i,int j) {return (i-1)*m+j;}
inline ll max(ll a,ll b) {return a>b?a:b;}
// inline void print(ll *a) { cout<<endl<<"fasdfas"; //调试输出
//     for(R i=0;i<=p;++i) printf("%lld ",a[i]); cout<<endl;
// }
// inline void print2(ll a[70][70]) { cout<<"eeewee";
//     for(R i=0;i<=p;++i,cout<<endl<<"      ") for(R j=0;j<=p;++j) printf("%lld ",a[i][j]);
// }
inline void mul1(ll a[70][70],ll b[70][70]) {
    register ll ret[70][70]; memset(ret,0,sizeof(ret));
    for(R i=0;i<=p;++i) for(R k=0;k<=p;++k) if(a[i][k]) for(R j=0;j<=p;++j)
        ret[i][j]+=a[i][k]*b[k][j];
    memcpy(a,ret,sizeof(ret));
}
inline void mul2(ll f[70],ll a[70][70]) {
    register ll ret[70]; memset(ret,0,sizeof(ret));
    for(R j=0;j<=p;++j) for(R k=0;k<=p;++k) 
        ret[j]+=f[k]*a[k][j];
    memcpy(f,ret,sizeof(ret));
}
inline void build() {
    for(R k=1;k<=60;e[k][0][0]=1,++k) for(R i=1;i<=n;++i) for(R j=1;j<=m;++j) {
        R x=a[i][j],p=cnt[i][j]; 
        if(b[x][p]>='0'&&b[x][p]<='9') {
            e[k][0][pos(i,j)]=b[x][p]-'0';
            e[k][pos(i,j)][pos(i,j)]=1;
        } else if(b[x][p]=='N'&&i>1) e[k][pos(i,j)][pos(i-1,j)]=1;
        else if(b[x][p]=='W'&&j>1) e[k][pos(i,j)][pos(i,j-1)]=1;
        else if(b[x][p]=='S'&&i<n) e[k][pos(i,j)][pos(i+1,j)]=1;
        else if(b[x][p]=='E'&&j<m) e[k][pos(i,j)][pos(i,j+1)]=1;
        cnt[i][j]=(p+1)%strlen(b[x]);
    } if(t>60) {memcpy(d,e[1],sizeof(e[1])); for(R i=2;i<=60;++i) mul1(d,e[i]);}
}
signed main() {
    n=g(),m=g(),t=g(),q=g(); p=n*m;
    for(R i=1;i<=n;++i) {
        scanf("%s",s+1); 
        for(R j=1;j<=m;++j) ch=s[j],a[i][j]=(ch^48)+1;
    } for(R i=1;i<=q;++i) scanf("%s",&b[i]);
    build(); f[0]=1; 
    for(R i=t/60;i;i>>=1,mul1(d,d)) if(i&1) mul2(f,d);
    for(R i=1,lim=t%60;i<=lim;++i) mul2(f,e[i]);
    for(R i=1;i<=p;++i) ans=max(ans,f[i]); printf("%lld\n",ans); 
    //while(1);
}

别颓废,至少看起来你懂了。2019.05.10

 

posted @ 2019-05-10 13:57  LuitaryiJack  阅读(207)  评论(0编辑  收藏  举报