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=,fix=; register char ch; while(!isdigit(ch=getchar())) fix=ch=='-'?-:fix;
do ret=ret*+(ch^); while(isdigit(ch=getchar())); return ret*fix;
}
int n,m,t,p,q;
int a[][],cnt[][];
ll f[],d[][],e[][][];
ll ans;
char b[][],s[],ch;
inline int pos(int i,int j) {return (i-)*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[][],ll b[][]) {
register ll ret[][]; memset(ret,,sizeof(ret));
for(R i=;i<=p;++i) for(R k=;k<=p;++k) if(a[i][k]) for(R j=;j<=p;++j)
ret[i][j]+=a[i][k]*b[k][j];
memcpy(a,ret,sizeof(ret));
}
inline void mul2(ll f[],ll a[][]) {
register ll ret[]; memset(ret,,sizeof(ret));
for(R j=;j<=p;++j) for(R k=;k<=p;++k)
ret[j]+=f[k]*a[k][j];
memcpy(f,ret,sizeof(ret));
}
inline void build() {
for(R k=;k<=;e[k][][]=,++k) for(R i=;i<=n;++i) for(R j=;j<=m;++j) {
R x=a[i][j],p=cnt[i][j];
if(b[x][p]>=''&&b[x][p]<='') {
e[k][][pos(i,j)]=b[x][p]-'';
e[k][pos(i,j)][pos(i,j)]=;
} else if(b[x][p]=='N'&&i>) e[k][pos(i,j)][pos(i-,j)]=;
else if(b[x][p]=='W'&&j>) e[k][pos(i,j)][pos(i,j-)]=;
else if(b[x][p]=='S'&&i<n) e[k][pos(i,j)][pos(i+,j)]=;
else if(b[x][p]=='E'&&j<m) e[k][pos(i,j)][pos(i,j+)]=;
cnt[i][j]=(p+)%strlen(b[x]);
} if(t>) {memcpy(d,e[],sizeof(e[])); for(R i=;i<=;++i) mul1(d,e[i]);}
}
signed main() {
n=g(),m=g(),t=g(),q=g(); p=n*m;
for(R i=;i<=n;++i) {
scanf("%s",s+);
for(R j=;j<=m;++j) ch=s[j],a[i][j]=(ch^)+;
} for(R i=;i<=q;++i) scanf("%s",&b[i]);
build(); f[]=;
for(R i=t/;i;i>>=,mul1(d,d)) if(i&) mul2(f,d);
for(R i=,lim=t%;i<=lim;++i) mul2(f,e[i]);
for(R i=;i<=p;++i) ans=max(ans,f[i]); printf("%lld\n",ans);
//while(1);
}

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

BZOJ 2973 石头游戏 矩乘加速递推的更多相关文章

  1. CH 3401 - 石头游戏 - [矩阵快速幂加速递推]

    题目链接:传送门 描述石头游戏在一个 $n$ 行 $m$ 列 ($1 \le n,m \le 8$) 的网格上进行,每个格子对应一种操作序列,操作序列至多有 $10$ 种,分别用 $0 \sim 9$ ...

  2. [模板][题解][Luogu1939]矩阵乘法加速递推(详解)

    题目传送门 题目大意:计算数列a的第n项,其中: \[a[1] = a[2] = a[3] = 1\] \[a[i] = a[i-3] + a[i - 1]\] \[(n ≤ 2 \times 10^ ...

  3. HDU 5950 - Recursive sequence - [矩阵快速幂加速递推][2016ACM/ICPC亚洲区沈阳站 Problem C]

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5950 Farmer John likes to play mathematics games with ...

  4. POJ3070 Fibonacci(矩阵快速幂加速递推)【模板题】

    题目链接:传送门 题目大意: 求斐波那契数列第n项F(n). (F(0) = 0, F(1) = 1, 0 ≤ n ≤ 109) 思路: 用矩阵乘法加速递推. 算法竞赛进阶指南的模板: #includ ...

  5. bzoj2004公交线路——DP+矩阵加速递推

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2004 求方案数,想到DP: 因为两个站间距离<=p,所以每p个站中所有车一定都会停靠至 ...

  6. HDU 1757 矩阵快速幂加速递推

    题意: 已知: 当x<10时:f(x)=x 否则:f(x) = a0 * f(x-1) + a1 * f(x-2) + a2 * f(x-3) + --+ a9 * f(x-10); 求:f(x ...

  7. 【csp模拟赛3】bridge.cpp--矩阵加速递推

    题目描述 穿越了森林,前方有一座独木桥,连接着过往和未来(连接着上一题和下一题...). 这座桥无限长. 小 Q 在独木桥上彷徨了.他知道,他只剩下了 N 秒的时间,每一秒的时间里,他会向 左或向右移 ...

  8. luogu题解 P1707 【刷题比赛】矩阵加速递推

    题目链接: https://www.luogu.org/problemnew/show/P1707 分析: 洛谷的一道原创题,对于练习矩阵加速递推非常不错. 首先我们看一下递推式: \(a[k+2]= ...

  9. CH3401 石头游戏(矩阵快速幂加速递推)

    题目链接:传送门 题目: 石头游戏 0x30「数学知识」例题 描述 石头游戏在一个 n 行 m 列 (≤n,m≤) 的网格上进行,每个格子对应一种操作序列,操作序列至多有10种,分别用0~9这10个数 ...

随机推荐

  1. BZOJ4676 Xor-Mul棋盘

    传送门 题目大意懒得写了,题目说的挺明白的了 题解 主要的难点在于异或意义下的最大值和很玄学,但不难发现这道题中让你定义的$D_{i,j}$只参与异或运算,所以我们可以逐位进行讨论.所以我们每一位就只 ...

  2. P1030 求先序排列

    题目描述 给出一棵二叉树的中序与后序排列.求出它的先序排列.(约定树结点用不同的大写字母表示,长度<=8). 输入输出格式 输入格式: 2行,均为大写字母组成的字符串,表示一棵二叉树的中序与后序 ...

  3. 三种 Failover 之 Client-Side Connect time Failover、Client-Side TAF、Service-Side TAF

    三种 Failover 之 Client-Side Connect time Failover.Client-Side TAF.Service-Side TAF 理论背景 Oracle  RAC 同时 ...

  4. 桥接以及Mercury MW54R中继

    家里连个路由器,一个是比较先进的TP-Link的TL-WR842N(100M),另外一个是比较古老的水星(Mercury) MW54R(54M),我们知道新的路由器都有WDS功能,方便作为副路由器(中 ...

  5. 洛谷【P1619】 解一元二次方程的烦恼

    我对模拟的理解:https://www.cnblogs.com/AKMer/p/9064018.html 题目传送门:https://www.luogu.org/problemnew/show/P16 ...

  6. Golang Channel用法简编

    转自:http://tonybai.com/2014/09/29/a-channel-compendium-for-golang/ 在进入正式内容前,我这里先顺便转发一则消息,那就是Golang 1. ...

  7. ES6学习之Symbol

    定义:一种新的原始数据类型,表示独一无二的值 const a = Symbol(); const b = Symbol("foo") //接收参数的Symbol,参数表示对Symb ...

  8. 什么是Nginx?为什么使用Nginx?

    源自 https://blog.csdn.net/yougoule/article/details/78186138 一.前言      为毛要用nginx服务器代理,不直接用tomcat 7.0,还 ...

  9. Web Form要“jquery”ScriptResourceMapping。请添加一个名为 jquery (区分大小写)的 ScriptResourceMapping。”的解决办法。

    1.先将aspnet.scriptmanager.jquery.dl 复制到bin  (网站根目录下的bin文件夹找不到,看看下面的图片中点击[显示所有文档])  文件夹下.   2.在网站根目录下s ...

  10. 使用MySQL客户端登录Ensemble数据库查询相关信息

    Ensemble公共MySQL数据库 对于大量数据和更详细的分析,Ensemble的MySQL服务器ensembldb.ensembl.org,useastdb.ensembl.org或asiadb. ...