我们看到了及其可怕的数据范围

这个样子都没有办法直接读入的数据范围应该怎么算

我们观察一下递推式\(f[i][j]=a*f[i][j]+b(j!=1)\)

\(f[i][1]=c*f[i-1][m]+d\)

转移非常简单,于是可以考虑一下矩阵乘法

如果我们将这个矩阵破坏成一个链,那么就会有这种形式的递推

连续推\(m\)次第一个柿子,之后再推一次第二个柿子,之后反复

重复上面的过程\(n\)次就好了

于是我们可以将连续转移\(m\)次一式的到的矩阵和第二个式子的转移矩阵乘起来,之后将这个矩阵再转移\(n\)次就是答案了

由于\(n,m\)是在太大了,我们又发现模数是质数,于是我们可以利用费马小定理来降幂

则有

\[A^{mod-1}\%mod=1
\]

所以

\[A^m\equiv A^{m\%(mod-1)}(\%\ mod )
\]

至于为什么要特判\(a=1\)

我怎么知道啊

#include<iostream>
#include<cstring>
#include<cstdio>
#define re register
#define maxn 5
#define LL long long
const LL mod=1000000007;
char N_[1000005],M_[1000005];
LL n,m,a,b,c,d;
inline LL read()
{
char c=getchar();
LL x=0;
while(c<'0'||c>'9') c=getchar();
while(c>='0'&&c<='9')
x=x*10%(mod-1)+c-48,c=getchar();
return x;
}
struct Mat
{
LL a[4][4],ans[4][4];
inline void did_ans()
{
LL mid[4][4];
for(re int i=1;i<=2;i++)
for(re int j=1;j<=2;j++)
mid[i][j]=ans[i][j],ans[i][j]=0;
for(re int i=1;i<=2;i++)
for(re int j=1;j<=2;j++)
for(re int p=1;p<=2;p++)
ans[i][j]=(ans[i][j]+(a[i][p]*mid[p][j])%mod)%mod;
}
inline void did_a()
{
LL mid[4][4];
for(re int i=1;i<=2;i++)
for(re int j=1;j<=2;j++)
mid[i][j]=a[i][j],a[i][j]=0;
for(re int i=1;i<=2;i++)
for(re int j=1;j<=2;j++)
for(re int p=1;p<=2;p++)
a[i][j]=(a[i][j]+(mid[i][p]*mid[p][j])%mod)%mod;
}
}R,L,K;
inline void mul(Mat &A,Mat &B)
{
LL mid[4][4];
for(re int i=1;i<=2;i++)
for(re int j=1;j<=2;j++)
mid[i][j]=A.ans[i][j],A.ans[i][j]=0;
for(re int i=1;i<=2;i++)
for(re int j=1;j<=2;j++)
for(re int p=1;p<=2;p++)
A.ans[i][j]=(A.ans[i][j]+(mid[i][p]*B.ans[p][j])%mod)%mod;
for(re int i=1;i<=2;i++)
for(re int j=1;j<=2;j++)
A.a[i][j]=A.ans[i][j];
}
int main()
{
scanf("%s%s",N_+1,M_+1);
a=read(),b=read(),c=read(),d=read();
int lenn=strlen(N_+1);
if(a==1)
{
for(re int i=1;i<=lenn;i++) n=n*10%mod+N_[i]-48;
n=(n-2+mod)%mod;
}
else
{
for(re int i=1;i<=lenn;i++) n=n*10%(mod-1)+N_[i]-48;
n=(n-2+mod-1)%(mod-1);
}
lenn=strlen(M_+1);
if(c==1)
{
for(re int i=1;i<=lenn;i++) m=m*10%mod+M_[i]-48;
m=(m-2+mod)%mod;
}
else
{
for(re int i=1;i<=lenn;i++) m=m*10%(mod-1)+M_[i]-48;
m=(m-2+mod-1)%(mod-1);
}
R.a[1][1]=R.ans[1][1]=1;
R.a[2][1]=R.ans[2][1]=b;
R.ans[2][2]=R.a[2][2]=a;
R.ans[1][2]=R.a[1][2]=0;
K=R;
while(m)
{
if(m&1) R.did_ans();
m>>=1;
R.did_a();
}
L.a[1][1]=L.ans[1][1]=1;
L.a[2][1]=L.ans[2][1]=d;
L.a[2][2]=L.ans[2][2]=c;
L.ans[1][2]=L.a[1][2]=0;
mul(L,R);
while(n)
{
if(n&1) L.did_ans();
n>>=1;
L.did_a();
}
printf("%lld",((L.ans[2][1]+L.ans[2][2])%mod*R.ans[2][2]%mod+R.ans[2][1])%mod);
return 0;
}

【[NOI2013]矩阵游戏】的更多相关文章

  1. bzoj 3240: [Noi2013]矩阵游戏 矩阵乘法+十进制快速幂+常数优化

    3240: [Noi2013]矩阵游戏 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 613  Solved: 256[Submit][Status] ...

  2. BZOJ 3240: [Noi2013]矩阵游戏

    3240: [Noi2013]矩阵游戏 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 1586  Solved: 698[Submit][Status ...

  3. BZOJ 3240([Noi2013]矩阵游戏-费马小定理【矩阵推论】-%*s-快速读入)

    3240: [Noi2013]矩阵游戏 Time Limit: 10 Sec   Memory Limit: 256 MB Submit: 123   Solved: 73 [ Submit][ St ...

  4. (十进制高速幂+矩阵优化)BZOJ 3240 3240: [Noi2013]矩阵游戏

    题目链接: http://www.lydsy.com/JudgeOnline/problem.php?id=3240 3240: [Noi2013]矩阵游戏 Time Limit: 10 Sec  M ...

  5. P1397 [NOI2013]矩阵游戏(递推)

    P1397 [NOI2013]矩阵游戏 一波化式子,$f[1][m]=a^{m-1}+b\sum_{i=0}^{m-2}a^i$,用快速幂+逆元求等比数列可以做到$logm$ 设$v=a^{m-1}, ...

  6. 【bzoj3240】 Noi2013—矩阵游戏

    http://www.lydsy.com/JudgeOnline/problem.php?id=3240 (题目链接) 题意$${F[1][1]=1}$$$${F[i][j]=a*F[i][j-1]+ ...

  7. BZOJ3240 [Noi2013]矩阵游戏

    本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000作者博客:http://www.cnblogs.com/ljh2000-jump/转 ...

  8. 3240: [Noi2013]矩阵游戏

    Description 婷婷是个喜欢矩阵的小朋友,有一天她想用电脑生成一个巨大的n行m列的矩阵(你不用担心她如何存储).她生成的这个矩阵满足一个神奇的性质:若用F[i][j]来表示矩阵中第i行第j列的 ...

  9. NOI2013矩阵游戏

    Description 婷婷是个喜欢矩阵的小朋友,有一天她想用电脑生成一个巨大的n行m列的矩阵(你不用担心她如何存储).她生成的这个矩阵满足一个神奇的性质:若用F[i][j]来表示矩阵中第i行第j列的 ...

  10. 洛谷P1397 [NOI2013]矩阵游戏

    矩阵快速幂+费马小定理 矩阵也是可以跑费马小定理的,但是要注意这个: (图是盗来的QAQ) 就是说如果矩阵a[i][i]都是相等的,那么就是mod p 而不是mod p-1了 #include< ...

随机推荐

  1. 《The Python Tutorial》——Errors and Exceptions 阅读笔记

    Errors and Exceptions 官方文档:https://docs.python.org/3.5/tutorial/errors.html python中所有的异常都继承自BaseExce ...

  2. LeetCode 257.二叉树所有路径(C++)

    给定一个二叉树,返回所有从根节点到叶子节点的路径. 说明: 叶子节点是指没有子节点的节点. 示例: 输入: 1 / \ 2 3 \ 5 输出: ["1->2->5", ...

  3. Android中改变Activity的不同icon:activity-alias

    Android设置title中的Icon有几种方法,介绍如下: 一种是直接在AndroidManifest.xml文件中设置android:icon属性,这种方法简单有效,应该算是我们最常用的设置Ic ...

  4. 九度oj题目1511:从尾到头打印链表

    题目1511:从尾到头打印链表 时间限制:1 秒 内存限制:128 兆 特殊判题:否 提交:6010 解决:1805 题目描述: 输入一个链表,从尾到头打印链表每个节点的值. 输入: 每个输入文件仅包 ...

  5. nyoj 1192——Salvation——————【搜索】

    Salvation 时间限制:1000 ms  |  内存限制:65535 KB 难度:3   描述 神秘瀑布镇是一个神秘的地方,那里有吸血鬼,狼人,巫师,二重身. Klaus(吸血鬼祖先) 为了利用 ...

  6. Java Collection.Set

    package 集合; /** * Set不包含重复元素 存储顺序和取出数据不一样 * * HashSet:它不保证set的迭代顺序,特别是它不保证该顺序恒久不变 * 底层是哈希表结构的 * Link ...

  7. 【Linux相识相知】计算机的组成、linux发行版和哲学思想、基础命令和目录结构(FHS)

    从今天开始,Frank将开始在博客上记录自己学习linux的点点滴滴,F初来乍到,还望各位大佬多多指教.本次博客的主要内容如下: 计算机基础:简要的描述了计算机的组成及其功能: linux初识:介绍了 ...

  8. lua load

    load (chunk [, chunkname [, mode [, env]]]) 加载一个代码块. 如果 chunk 是一个字符串,代码块指这个字符串. 如果 chunk 是一个函数, load ...

  9. scss-注释

    在scss中有两种注释方式 原生css的注释多行注释: /* *  注释的内容 */ 单行注释:// 注释内容一致延续到行末. 在尽可能的情况下,多行注释会被保留在输出的CSS中,而单行注释会被删除.

  10. 最新机动车行驶证模板PSD可编辑分层文件下载

    机动车行驶证PSD模板下载地址: http://www.qijieworld.com/thread-1834752-1-1.html 模板为psd格式,内容可编辑修改,需使用 Photoshop CS ...