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. Gym - 101196G :That's One Hanoi-ed Teacher (递推)

    题意:给定当前汉诺塔的状态,问还有多少步走完,不合法输出“No”. 思路:显然可以一层一层试探下去的.我们设三个柱子为“起始”,“中转”,“终点”,当前状态的最大的盘子不可能在中转站处:如果在起始站, ...

  2. CQOI2018做题记录

    T1.破解D-H协议 传送门 这个题就是BSGS的板子题-- 然后这里补充一点嘛,就是第二重循环的枚举范围.我们是在枚举\(a^{tm-y}\),把tm换成i,这个的最大值就是\(i - (m - 1 ...

  3. bzoj 2648 SJY摆棋子 —— K-D树

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2648 学习资料:https://blog.csdn.net/zhl30041839/arti ...

  4. MongoDB3.2.8创建初始用户

    启动MongoDB前需要关闭配置文件中的auth选项,否则不能创建用户 首先创建用户管理用户 use admin db.createUser({user:'admin',pwd:'123456', r ...

  5. 2006浙大火星A+B

    题目描述:     读入两个不超过25位的火星正整数A和B,计算A+B.需要注意的是:在火星上,整数不是单一进制的,第n位的进制就是第n个素数.例如:地球上的10进制数2,在火星上记为“1,0”,因为 ...

  6. 【转】 Pro Android学习笔记(三五):Menu(6):XML方式 & PopUp菜单

    目录(?)[-] 利用XML创建菜单 XML的有关属性 onClick事件 Pop-up菜单 利用XML创建菜单 在代码中对每个菜单项进行设置,繁琐且修改不灵活,不能适配多国语言的要求,可以利用资源进 ...

  7. 【转】 Pro Android学习笔记(二九):用户界面和控制(17):include和merge

    目录(?)[-] xml控件代码重用include xml控件代码重用merge 横屏和竖屏landsacpe portrait xml控件代码重用:include 如果我们定义一个控件,需要在不同的 ...

  8. nmp部署(Nginx Mariadb Php-fpm)

    #主机:192.168.2.129(mini2) 既是php主机,也是数据库主机#yum install -y php php-fpm php-mysql mariadb-server[root@~ ...

  9. linux日常管理-sar工具

    查看网卡瓶颈 查看网卡流量 默认10分钟一次 查看实时流量  每秒钟显示一次 显示5次 网卡有 lo eth0   主要看eth0外网  rxbyt/s 进网口和 txbyt/s出网口 带宽看txby ...

  10. React中state与props介绍与比较

    一.state 1.state的作用 state是React中组件的一个对象.React把用户界面当做是状态机,想象它有不同的状态然后渲染这些状态,可以轻松让用户界面与数据保持一致. React中,更 ...