【BZOJ4688】One-Dimensional(矩阵快速幂)

Description

考虑一个含有 N 个细胞的一维细胞自动机。细胞从 0 到 N-1 标号。每个细胞有一个被表示成一个小于 M 的非负整数的状态。细胞的状态会在每个整数时刻发生骤变。我们定义 S(i,t) 表示第 i 个细胞在时刻 t 的状态。在时刻 t+1 的状态被表示为 S(i,t+1)=(A×S(i-1,t)+B×S(i,t)+C×S(i+1,t) ) mod M ,其中 A,B,C 是给定的非负整数。对于 i<0 或 N≤i ,我们定义 S(i,t)=0 。给定一个自动机的定义和其细胞在时刻 0 的初始状态,你的任务是计算时刻 T 时每个细胞的状态。

Input

输入包含多组测试数据。每组数据的第一行包含六个整数 N,M,A,B,C,T ,满足 0 < N≤50,0 < M≤1000,0≤A,B,C < M,0≤T≤〖10〗^9 。第二行包含 N 个小于 M 的非负整数,依次表示每个细胞在时刻 0 的状态。输入以六个零作为结束。

Output

对于每组数据,输出N个小于M的非负整数,每两个相邻的数字之间用一个空格隔开,表示每个细胞在时刻T的状态。

Sample Input

5 4 1 3 2 0

0 1 2 0 1

5 7 1 3 2 1

0 1 2 0 1

5 13 1 3 2 11

0 1 2 0 1

5 5 2 0 1 100

0 1 2 0 1

6 6 0 2 3 1000

0 1 2 0 1 4

20 1000 0 2 3 1000000000

0 1 2 0 1 0 1 2 0 1 0 1 2 0 1 0 1 2 0 1

30 2 1 0 1 1000000000

0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0

30 2 1 1 1 1000000000

1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0

30 5 2 3 1 1000000000

0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0

0 0 0 0 0 0

Sample Output

0 1 2 0 1

2 0 0 4 3

2 12 10 9 11

3 0 4 2 1

0 4 2 0 4 4

0 376 752 0 376 0 376 752 0 376 0 376 752 0 376 0 376 752 0 376

1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0

1 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0

1 1 3 2 2 2 3 3 1 4 3 1 2 3 0 4 3 3 0 4 2 2 2 2 1 1 2 1 3 0

题解:矩阵快速幂。好久没写,练个手。

#include<iostream>
#include<stdio.h>
#include<algorithm>
#include<string.h>
#include<math.h>
#define ll long long
#define inf 0x7f7f7f7f
#define N 55
using namespace std;
int read()
{
    int x=0,f=1;
    char c=getchar();
    while(c<'0' || c>'9') {if(c=='-') f=-1;c=getchar();}
    while(c<='9' && c>='0') {x=x*10+c-'0';c=getchar();}
    return x*f;
}
int t,n,mod,a,b,c;
struct mat
{
    int d[N][N];
    int x,y;
    mat(){}
    mat(int a,int b):x(a),y(b){memset(d,0,sizeof(d));}//normal
    mat(int a):x(a),y(a)//unitmat
    {
        memset(d,0,sizeof(d));
        for(int i=0;i<x;i++) d[i][i]=1;
    }
    mat operator *(const mat f)
    {
        mat t(x,f.y);
        for(int i=0;i<x;i++)
        for(int j=0;j<y;j++)
        for(int k=0;k<f.y;k++) t.d[i][k]=(t.d[i][k]+d[i][j]*f.d[j][k])%mod;
        return t;
    }
    mat mpow(int n)
    {
        mat t(x),now=*this;
        while(n)
        {
            if(n&1) t=t*now;
            now=now*now;
            n>>=1;
        }
        return t;
    }
}A,B;
int main()
{
    while(1)
    {
        n=read(),mod=read(),a=read(),b=read(),c=read(),t=read();
        if(!n) break;
        A=mat(1,n);B=mat(n,n);
        for(int i=0;i<n;i++) A.d[0][i]=read();
        for(int i=1;i<n;i++) B.d[i-1][i]=a;
        for(int i=0;i<n;i++) B.d[i][i]=b;
        for(int i=0;i<n-1;i++) B.d[i+1][i]=c;
        B=B.mpow(t);
        A=A*B;
        printf("%d",A.d[0][0]);
        for(int i=1;i<n;i++) printf(" %d",A.d[0][i]);
        printf("\n");
    }
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值