题目

一个细胞自动机包含 $n$ 个格子,每个格子的取值为 $0 \sim m-1$。给定距离 $d$,则每次操作是将每个格子的值变为到它的距离不超过 $d$ 的所有格子的在操作之前的值的和除以 $m$ 的余数。给出 $n, m, d, k$ 和自动机各个格子的初始值。你的任务是计算 $k$ 次操作以后各格子的值。($1 \leq n\leq 500, 1 \leq m\leq 10^6, 0 \leq d\leq n/2, 1\leq k\leq 10^7$).

分析

如果我们把 $t$ 次操作以后的各格子值写成列向量 $v_t$,不难发现 $v_{t+1}$ 的每一维都是 $v_t$ 中各维的线性组合,其中的加法和乘法都是在模 $m$ 的剩余系中完成。

每次操作相当于乘以一个 $n \times n $ 矩阵,直接使用矩阵快速幂的复杂度为 $O(n^3logk)$,

由于这里的矩阵比较特殊,是循环矩阵(从第二行开始每一行都是上一行循环右移),

可以证明,两个循环矩阵的乘积仍然为循环矩阵。

因此在存储时只需保存第一行,而计算矩阵乘法时也只需算出第一行即可。这样,矩阵乘法的时间复杂度降为 $O(n^2)$。总时间降为 $O(n^2log k)$,可以承受。

用FFT优化的话可做到 $0(nlognlogk)$.  

$$\begin{bmatrix}
1 & 1 & 0 & 0 & 1\\
1 & 1 & 1 & 0 & 0 \\
0 &  1&  1& 1 & 0\\
0 & 0 & 1 & 1 & 1\\
1 & 0 & 0 & 1 & 1
\end{bmatrix} \times
\begin{bmatrix}
1 & 1 & 0 & 0 & 1\\
1 & 1 & 1 & 0 & 0 \\
0 &  1&  1& 1 & 0\\
0 & 0 & 1 & 1 & 1\\
1 & 0 & 0 & 1 & 1
\end{bmatrix} =
\begin{bmatrix}
3 & 2 & 1 & 1 & 2\\
2 & 3 & 2 & 1 & 1\\
1 & 2 & 3 & 2 & 1\\
1 & 1 & 2 & 3 & 2\\
2 & 1 & 1 & 2 & 3
\end{bmatrix}$$

#include<cstdio>
#include<cstring>
using namespace std; typedef long long ll;
const int maxn = +;
struct matrix
{
int n;
ll mat[maxn];
matrix(){
memset(mat, , sizeof(mat));
}
};
ll n, p, d, k;
ll a[maxn]; matrix mul(matrix A, matrix B) //矩阵相乘,这里A=B,且都是n x n的方阵
{
matrix ret;
ret.n = A.n;
for(int i = ;i < A.n;i++)
for(int j = ;j < B.n;j++) ret.mat[i] = (ret.mat[i] + A.mat[j] * B.mat[(j-i+A.n)%A.n]) % p;
return ret;
} matrix mpow(matrix A, int n)
{
matrix ret;
ret.n = A.n;
ret.mat[]=;
while(n)
{
if(n & ) ret = mul(ret, A);
A = mul(A, A);
n >>= ;
}
return ret;
} int main()
{
while(scanf("%lld%lld%lld%lld", &n, &p,&d, &k) == )
{
for(int i = ;i < n;i++) scanf("%lld", &a[i]);
matrix A;
A.n = n;
for(int i = ;i <= d;i++) A.mat[i] = ;
for(int i = n-d; i < n;i++) A.mat[i] = ; A = mpow(A, k); for(int i = ;i < A.n;i++)
{
ll tmp = ;
for(int j = ;j < A.n;j++) tmp = (tmp + A.mat[(j-i+n) % n] * a[j]) % p;
printf("%lld%c", tmp, i == n- ? '\n' : ' ');
}
}
return ;
}

记得开 long long !!

参考链接: https://vjudge.net/status/#un=&OJId=UVALive&probNum=3704&res=0&orderBy=run_id&language=

最新文章

  1. Spring WebService入门
  2. 《连载 | 物联网框架ServerSuperIO教程》- 13.自定义视图显示接口开发,满足不同的显示需求
  3. dict和set
  4. java 15-1 Collection集合的概述以及小功能介绍
  5. Spring整合Quartz实现持久化、动态设定时间
  6. hadoop优点和缺点
  7. css3 去掉点击高光(移动端)
  8. NGUI学习笔记(一):官方视频学习记录
  9. PHP学习笔记13淘宝接口开发一例(tmall.items.discount.search),PHP
  10. [产值分析]生产部KPI考核之产值分析
  11. java的static关键字 – Break易站
  12. Google学术搜索镜像网站搜集
  13. [Swift]LeetCode268. 缺失数字 | Missing Number
  14. 机器学习 GBDT+xgboost 决策树提升
  15. python_项目_ATM和购物商城的程序
  16. idea 项目转 eclipse项目
  17. 第二节 JavaScript基础
  18. 阿里云免费申请https证书
  19. Inno Setup自定义安装界面脚本
  20. Java实现继承过程概述

热门文章

  1. JVM的内存分配垃圾回收策略
  2. Git手册(一):基本操作
  3. Django模型层之单表操作
  4. 日志之slf4j和logback日志系统(二)
  5. 上传自己的 NuGet 包
  6. [jsp学习笔记] jsp过滤器
  7. ubuntu18.04使用vscode报pylint is not install错误
  8. 【转载】C#使用Math.Ceiling方法对计算结果向上取整操作
  9. 页面、 ajax 、mock
  10. html,css,js(包含简单的 ES6语法) 实现 简单的音乐盒