BZOJ 3240([Noi2013]矩阵游戏-费马小定理【矩阵推论】-%*s-快速读入)
3240: [Noi2013]矩阵游戏
Time Limit: 10 Sec   Memory Limit: 256 MB 
 
  Submit: 123  
  Solved: 73
  
 [
  Submit][
  Status]
Description
婷婷是个喜欢矩阵的小朋友,有一天她想用电脑生成一个巨大的n行m列的矩阵(你不用担心她如何存储)。她生成的这个矩阵满足一个神奇的性质:若用F[i][j]来表示矩阵中第i行第j列的元素,则F[i][j]满足下面的递推式:
F[1][1]=1
 F[i,j]=a*F[i][j-1]+b (j!=1)
 F[i,1]=c*F[i-1][m]+d (i!=1)
 递推式中a,b,c,d都是给定的常数。
现在婷婷想知道F[n][m]的值是多少,请你帮助她。由于最终结果可能很大,你只需要输出F[n][m]除以1,000,000,007的余数。
Input
一行有六个整数n,m,a,b,c,d。意义如题所述
Output
包含一个整数,表示F[n][m]除以1,000,000,007的余数
Sample Input
Sample Output
HINT
样例中的矩阵为:
1 4 7 10
26 29 32 35
76 79 82 85
Source
NOI赛场上拿了70分(没用费马小定理)
PS:实际原因是没算复杂度囧。。。。
好吧。。。这题让我们回顾一下费马小定理:
a^b mod (phi(p))=1 (mod p)
所以可以mod ,特判a=c=1的特殊情况
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<functional>
#include<iostream>
#include<cmath>
#include<cctype>
#include<ctime>
using namespace std;
#define For(i,n) for(int i=1;i<=n;i++)
#define Fork(i,k,n) for(int i=k;i<=n;i++)
#define Rep(i,n) for(int i=0;i<n;i++)
#define ForD(i,n) for(int i=n;i;i--)
#define RepD(i,n) for(int i=n;i>=0;i--)
#define Forp(x) for(int p=pre[x];p;p=next[p])
#define Lson (x<<1)
#define Rson ((x<<1)+1)
#define MEM(a) memset(a,0,sizeof(a));
#define MEMI(a) memset(a,127,sizeof(a));
#define MEMi(a) memset(a,128,sizeof(a));
#define INF (2139062143)
#define F (1000000007)
#define MAXN (1000000+10)
long long mul(long long a,long long b){return (a*b)%F;}
long long add(long long a,long long b){return (a+b)%F;}
long long sub(long long a,long long b){return (a-b+(a-b)/F*F+F)%F;}
typedef long long ll;
ll a,b,c,d,n,m,phiF;
ll getint(char s[]) //_dec_1
{
char c;
ll x=0;
for(int i=0,c=s[i];s[i];c=s[++i])
{
x=(x*10+c-48)%phiF;
}
x=(x-1+phiF)%phiF;
return x;
}
struct M
{
int n,m;
ll a[3][3];
M(){n=m=2;MEM(a);}
M(ll a1,ll a2,ll b1,ll b2){n=m=2;MEM(a) a[1][1]=a1,a[1][2]=a2,a[2][1]=b1,a[2][2]=b2; }
friend M operator*(M a,M b)
{
M c;
For(k,2)
For(i,2)
For(j,2)
c.a[i][j]=(c.a[i][j]+a.a[i][k]*b.a[k][j])%F;
/*
For(i,2)
{
For(j,2) cout<<c.a[i][j]<<' ';cout<<endl;
}*/
return c;
}
void make_I()
{
n=m=2;MEM(a)
For(i,n) a[i][i]=1;
}
}A,B,C,D,I;
void print(M a)
{
For(i,2)
{
For(j,2) cout<<a.a[i][j]<<' ';cout<<endl;
}
}
M pow2(M a,ll b)
{
M c=I;
static bool a2[MAXN];
int n=0;while (b) a2[++n]=b&1,b>>=1;
For(i,n)
{
if (a2[i]) c=c*a;
a=a*a;
}
return c;
}
char s1[MAXN],s2[MAXN];
int main()
{
scanf("%s%s",s1,s2);
scanf("%lld%lld%lld%lld",&a,&b,&c,&d);
if (a==1&&c==1) phiF=F;else phiF=F-1;
n=getint(s1);m=getint(s2);
A=M(a,b,0,1);B=M(c,d,0,1);I=M(1,0,0,1);
C=pow2(A,m);//print(C);
D=B*C;//print(D);
D=pow2(D,n);
D=C*D;//print(D);
cout<<(D.a[1][2]+D.a[1][1])%F<<endl; return 0;
}
顺便再附一份读文件的特殊写法:
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<functional>
#include<iostream>
#include<cmath>
#include<cctype>
#include<ctime>
using namespace std;
#define For(i,n) for(int i=1;i<=n;i++)
#define Fork(i,k,n) for(int i=k;i<=n;i++)
#define Rep(i,n) for(int i=0;i<n;i++)
#define ForD(i,n) for(int i=n;i;i--)
#define RepD(i,n) for(int i=n;i>=0;i--)
#define Forp(x) for(int p=pre[x];p;p=next[p])
#define Lson (x<<1)
#define Rson ((x<<1)+1)
#define MEM(a) memset(a,0,sizeof(a));
#define MEMI(a) memset(a,127,sizeof(a));
#define MEMi(a) memset(a,128,sizeof(a));
#define INF (2139062143)
#define F (1000000007)
#define MAXN (1000000+10)
long long mul(long long a,long long b){return (a*b)%F;}
long long add(long long a,long long b){return (a+b)%F;}
long long sub(long long a,long long b){return (a-b+(a-b)/F*F+F)%F;}
typedef long long ll;
ll a,b,c,d,n,m,phiF;
ll getint() //_dec_1
{
char c;
while (c=getchar(),(!isdigit(c)));
//cout<<(int)c;
ll x=c-48;
while (c=getchar(),(isdigit(c)))
{
x=(x*10+c-48)%phiF;
}
x=(x-1+phiF)%phiF;
return x;
}
struct M
{
int n,m;
ll a[3][3];
M(){n=m=2;MEM(a);}
M(ll a1,ll a2,ll b1,ll b2){n=m=2;MEM(a) a[1][1]=a1,a[1][2]=a2,a[2][1]=b1,a[2][2]=b2; }
friend M operator*(M a,M b)
{
M c;
For(k,2)
For(i,2)
For(j,2)
c.a[i][j]=(c.a[i][j]+a.a[i][k]*b.a[k][j])%F;
/*
For(i,2)
{
For(j,2) cout<<c.a[i][j]<<' ';cout<<endl;
}*/
return c;
}
void make_I()
{
n=m=2;MEM(a)
For(i,n) a[i][i]=1;
}
}A,B,C,D,I;
void print(M a)
{
For(i,2)
{
For(j,2) cout<<a.a[i][j]<<' ';cout<<endl;
}
}
M pow2(M a,ll b)
{
M c=I;
static bool a2[MAXN];
int n=0;while (b) a2[++n]=b&1,b>>=1;
For(i,n)
{
if (a2[i]) c=c*a;
a=a*a;
}
return c;
} int main()
{
freopen("matrix.in","r",stdin);
freopen("matrix.out","w",stdout);
scanf("%*s%*s");
scanf("%lld%lld%lld%lld",&a,&b,&c,&d);
if (a==1&&c==1) phiF=F;else phiF=F-1;
freopen("matrix.in","r",stdin);
n=getint();m=getint();
A=M(a,b,0,1);B=M(c,d,0,1);I=M(1,0,0,1);
C=pow2(A,m);//print(C);
D=B*C;//print(D);
D=pow2(D,n);
D=C*D;//print(D);
cout<<(D.a[1][2]+D.a[1][1])%F<<endl; return 0;
}
BZOJ 3240([Noi2013]矩阵游戏-费马小定理【矩阵推论】-%*s-快速读入)的更多相关文章
- BZOJ 3240 [Noi2013]矩阵游戏 ——费马小定理 快速幂
		
发现是一个快速幂,然而过不去. 怎么办呢? 1.十进制快速幂,可以用来练习卡时. 2.费马小定理,如果需要乘方的地方,可以先%(p-1)再计算,其他地方需要%p,所以需要保存两个数. 然后就是分类讨论 ...
 - bzoj3240 [Noi2013]矩阵游戏——费马小定理+推式子
		
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3240 n 和 m 太过巨大,不难想到应该用费马小定理什么的来缩小范围: 总之就是推式子啦,看 ...
 - 【bzoj5118】Fib数列2  费马小定理+矩阵乘法
		
题目描述 Fib定义为Fib(0)=0,Fib(1)=1,对于n≥2,Fib(n)=Fib(n-1)+Fib(n-2) 现给出N,求Fib(2^n). 输入 本题有多组数据.第一行一个整数T,表示数据 ...
 - HDOJ 4549 M斐波那契数列 费马小定理+矩阵高速幂
		
MF( i ) = a ^ fib( i-1 ) * b ^ fib ( i ) ( i>=3) mod 1000000007 是质数 , 依据费马小定理 a^phi( p ) = 1 ( ...
 - HDU 4549 (费马小定理+矩阵快速幂+二分快速幂)
		
M斐波那契数列 Time Limit: 1000MS Memory Limit: 32768KB 64bit IO Format: %I64d & %I64u Submit Statu ...
 - Fib数列2 费马小定理+矩阵乘法
		
题解: 费马小定理 a^(p-1)=1(mod p) 这里推广到矩阵也是成立的 所以我们可以对(2^n)%(p-1) 然后矩阵乘法维护就好了 模数较大使用快速乘
 - bzoj5118: Fib数列2(费马小定理+矩阵快速幂)
		
题目大意:求$fib(2^n)$ 就是求fib矩阵的(2^n)次方%p,p是质数,根据费马小定理有 注意因为模数比较大会爆LL,得写快速乘法... #include<bits/stdc++.h& ...
 - HDOJ 5667 Sequence//费马小定理 矩阵快速幂
		
题目:http://acm.hdu.edu.cn/showproblem.php?pid=5667 题意:如题给了一个函数式,给你a,b,c,n,p的值,叫你求f(n)%p的值 思路:先对函数取以a为 ...
 - 【费马小定理+矩阵快速幂】HDU4549——M斐波那契数列
		
[题目大意] M斐波那契数列F[n]是一种整数数列,它的定义如下:F[0] = aF[1] = bF[n] = F[n-1] * F[n-2] ( n > 1 )现在给出a, b, n,求出F[ ...
 
随机推荐
- POJ 2115 模线性方程 ax=b(mod n)
			
/* (x*c+a)%(2^k)==b →(x*c)%(2^k)==b-a 满足定理: 推论1:方程ax=b(mod n)对于未知量x有解,当且仅当gcd(a,n) | b. 推论2:方程ax=b(m ...
 - eclipse中tomcat启动项目 修改java代码不重启服务
			
1.双击tomcat 2.选择modules 3.选中项目点击edit 4.去掉勾.去除auto reloading enabled 的选中 ,点击OK,
 - centos 挂载windows共享目录
			
su (获取root权限) yum install samba 安装samba (其实我们只用到samba里面的winbind以便我们能够用windows机器的名称找到该机器的网络地址,在下面叙述的过 ...
 - Android 开发笔记“浅谈DDMS视图”
			
DDMS 的全称是Dalvik Debug Monitor Service,即Dalvik调试监控服务,是一个可视化的调试监控工具.它主要是对系统运行后台日志的监控,还有系统线程,模拟器状态的监控.此 ...
 - 高质量程序设计指南C/C++语言——malloc/free使用要点
 - Mathematica
			
Mathematica是一款科学计算软件,很好地结合了数值和符号计算引擎.图形系统.编程语言.文本系统.和与其他应用程序的高级连接.很多功能在相应领域内处于世界领先地位,它也是使用最广泛的数学软件之一 ...
 - PSAM卡
			
PSAM卡 终端安全控制模块,符合<中国金融集成电路(IC卡)PSAM卡规范>, 包括普通PSAM卡和高速PSAM卡. PSAM符合以下标准及规范: 识别卡,带触点的集成电路卡标 ...
 - CMake 教程
			
CMake是一个跨平台的程序构建工具,比如起自己编写Makefile方便很多. 介绍:http://baike.baidu.com/view/1126160.htm 本文件不介绍CMake的基本语法, ...
 - kinect for windows - SkeletonBasics-D2D详解之一
			
之前的文章介绍了深度图的获取,但是深度图只是提供了一些数据,这些数据给上层应用使用还是偏底层一些,我们希望在这个基础上,获取一些信息,比如手挥动,人跑步,或者运动等等,那么这个文章开始我们来讲述kin ...
 - Eclipse 代码提示不显示的问题
			
在备份Android SDK和Eclipse的时候出了问题,然后只能重新下载,SDK下得我都要吐血了,超慢.然后,我发现Eclipse有了新版本的,于是就下载了个新版的,结果出了问题了.所有的Andr ...