NOI2013矩阵游戏
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
3 4 1 3 2 6
Sample Output
85
HINT
样例中的矩阵为:
1 4 7 10
26 29 32 35
76 79 82 85
题解:
不看数据范围的话这就是到水题……
前10个点很好过,普通的二进制快速幂就可以
后面10个点如果还用原来的方法,需要涉及高精度除以单精度,复杂度是O(len)的
所以一种新的快速幂诞生了!-----十进制快速幂!
(转)
-------------------------------------------------------------------------------------------------------------------------------------------------------------------------
再举个例子吧...
const p=;
type matrix=array[..,..] of longint;
var a,b,c:matrix;
i,n,m,a1,a2,b1,b2:longint;
procedure mul(var x,y,z:matrix);
var t:matrix;
i,j,k:longint;
begin
fillchar(t,sizeof(t),);
for i:= to do
for j:= to do
for k:= to do
t[i,j]:=(t[i,j]+x[i,k]*y[k,j]) mod p;
z:=t;
end;
procedure ksm(cs:longint);
begin
while cs> do
begin
if cs and = then mul(a,b,b);
cs:=cs>>;
mul(a,a,a);
end;
end;
procedure init;
begin
readln(n,m,a1,b1,a2,b2);
end;
procedure main;
begin
a[,]:=;a[,]:=b1;a[,]:=;a[,]:=a1;
for i:= to do b[i,i]:=;
ksm(m-);
c:=b;
b[,]:=;b[,]:=b2;b[,]:=;b[,]:=a2;
mul(c,b,a);
fillchar(b,sizeof(b),);
for i:= to do b[i,i]:=;
ksm(n-);
mul(b,c,b);
writeln((b[,]+b[,]) mod p);
end;
begin
init;
main;
end.
代码2:80分 十进制快速幂(后四个点TLE)
const p=;
type matrix=array[..,..] of longint;
arrtype=array[..] of longint;
var a,b,c:matrix;
mm:array[..] of matrix;
i,a1,a2,b1,b2:longint;
n,m:arrtype;
ch:char;
procedure mul(var x,y,z:matrix);
var t:matrix;
i,j,k:longint;
begin
fillchar(t,sizeof(t),);
for i:= to do
for j:= to do
for k:= to do
t[i,j]:=(t[i,j]+x[i,k]*y[k,j]) mod p;
z:=t;
end;
procedure pow10(s:arrtype);
var i,j:longint;t:matrix;
begin
dec(s[s[]]);i:=s[];while s[i]< do begin inc(s[i],);dec(s[i-],);dec(i);end;
fillchar(mm,sizeof(mm),);
for i:= to do mm[,i,i]:=;
for i:= to do mul(mm[i-],a,mm[i]);
fillchar(b,sizeof(b),);
for i:= to do b[i,i]:=;
for i:= to s[] do
begin
mul(b,mm[s[i]],b);
if i=s[] then break;
t:=b;
for j:= to do mul(b,t,b);
end;
end;
procedure init;
begin
read(ch);n[]:=;
while ch<>' ' do
begin
inc(n[]);n[n[]]:=ord(ch)-ord('');
read(ch);
end;
read(ch);m[]:=;
while ch<>' ' do
begin
inc(m[]);m[m[]]:=ord(ch)-ord('');
read(ch);
end;
readln(a1,b1,a2,b2);
end;
procedure main;
begin
a[,]:=;a[,]:=b1;a[,]:=;a[,]:=a1;
pow10(m);
c:=b;
b[,]:=;b[,]:=b2;b[,]:=;b[,]:=a2;
mul(c,b,a);
pow10(n);
mul(b,c,b);
writeln((b[,]+b[,]) mod p);
end;
begin
init;
main;
end.
代码3:80分 考虑把代码二中 ans^10换成二进制快速幂(依然TLE)
const p=;
type matrix=array[..,..] of int64;
arrtype=array[..] of longint;
var a,b,c:matrix;
mm:array[..] of matrix;
i,a1,a2,b1,b2:longint;
n,m:arrtype;
ch:char;
procedure mul(var x,y,z:matrix);
var t:matrix;
i,j,k:longint;
begin
fillchar(t,sizeof(t),);
for i:= to do
for j:= to do
for k:= to do
t[i,j]:=(t[i,j]+x[i,k]*y[k,j]) mod p;
z:=t;
end;
procedure pow10(s:arrtype);
var i,j,cs:longint;tmp:matrix;
begin
dec(s[s[]]);i:=s[];while s[i]< do begin inc(s[i],);dec(s[i-],);dec(i);end;
fillchar(mm,sizeof(mm),);
for i:= to do mm[,i,i]:=;
for i:= to do mul(mm[i-],a,mm[i]);
fillchar(b,sizeof(b),);
for i:= to do b[i,i]:=;
for i:= to s[] do
begin
mul(b,mm[s[i]],b);
if i=s[] then break;
tmp:=b;
fillchar(b,sizeof(b),);
for j:= to do b[j,j]:=;
cs:=;
while cs> do
begin
if cs and = then mul(tmp,b,b);
cs:=cs>>;
mul(tmp,tmp,tmp);
end;
end;
end;
procedure init;
begin
read(ch);n[]:=;
while ch<>' ' do
begin
inc(n[]);n[n[]]:=ord(ch)-ord('');
read(ch);
end;
read(ch);m[]:=;
while ch<>' ' do
begin
inc(m[]);m[m[]]:=ord(ch)-ord('');
read(ch);
end;
readln(a1,b1,a2,b2);
end;
procedure main;
begin
a[,]:=;a[,]:=b1;a[,]:=;a[,]:=a1;
pow10(m);
c:=b;
b[,]:=;b[,]:=b2;b[,]:=;b[,]:=a2;
mul(c,b,a);
pow10(n);
mul(b,c,b);
writeln((b[,]+b[,]) mod p);
end;
begin
init;
main;
end.
代码4:受兰多夫87的影响,考虑修改矩阵乘法,应为第一列始终是不变的 (为什么还是TLE啊…………)
const p=;
type matrix=array[..,..] of int64;
arrtype=array[..] of longint;
var a,b,c:matrix;
mm:array[..] of matrix;
i,a1,a2,b1,b2:longint;
n,m:arrtype;
ch:char;
procedure mul(var x,y,z:matrix);
var t:matrix;
i,j,k:longint;
begin
fillchar(t,sizeof(t),);
t[,]:=;t[,]:=;
for i:= to do
for k:= to do
t[i,]:=(t[i,]+x[i,k]*y[k,]) mod p;
z:=t;
end;
procedure pow10(s:arrtype);
var i,j,cs:longint;tmp:matrix;
begin
dec(s[s[]]);i:=s[];while s[i]< do begin inc(s[i],);dec(s[i-],);dec(i);end;
fillchar(mm,sizeof(mm),);
for i:= to do mm[,i,i]:=;
for i:= to do mul(mm[i-],a,mm[i]);
fillchar(b,sizeof(b),);
for i:= to do b[i,i]:=;
for i:= to s[] do
begin
mul(b,mm[s[i]],b);
if i=s[] then break;
tmp:=b;
fillchar(b,sizeof(b),);
for j:= to do b[j,j]:=;
cs:=;
while cs> do
begin
if cs and = then mul(tmp,b,b);
cs:=cs>>;
mul(tmp,tmp,tmp);
end;
end;
end;
procedure init;
begin
read(ch);n[]:=;
while ch<>' ' do
begin
inc(n[]);n[n[]]:=ord(ch)-ord('');
read(ch);
end;
read(ch);m[]:=;
while ch<>' ' do
begin
inc(m[]);m[m[]]:=ord(ch)-ord('');
read(ch);
end;
readln(a1,b1,a2,b2);
end;
procedure main;
begin
a[,]:=;a[,]:=b1;a[,]:=;a[,]:=a1;
pow10(m);
c:=b;
b[,]:=;b[,]:=b2;b[,]:=;b[,]:=a2;
mul(c,b,a);
pow10(n);
mul(b,c,b);
writeln((b[,]+b[,]) mod p);
end;
begin
init;
main;
end.
代码5:我认为是我写丑了 膜拜兰多夫87吧
const p=;
type matrix=array[..] of int64;
arrtype=array[..] of longint;
var x,y:matrix;
z:array[..] of matrix;
i,a,b,c,d:longint;
n,m:arrtype;
ch:char;
operator *(a,b:matrix)c:matrix;
begin
c[]:=a[]*b[] mod p;
c[]:=(b[]*a[]+b[]) mod p;
end;
function f(x:matrix;var a:arrtype):matrix;
var i:longint;y:matrix;
begin
dec(a[a[]]);i:=a[];
while a[i]< do begin inc(a[i],);dec(a[i-]);dec(i);end;
z[,]:=;z[,]:=;
for i:= to do z[i]:=z[i-]*x;
x:=z[];
for i:= to a[] do
begin
y:=x*x;
x:=x*y*y;
x:=x*x;
x:=x*z[a[i]];
end;
exit(x);
end;
procedure init;
begin
read(ch);n[]:=;
while ch<>' ' do
begin
inc(n[]);n[n[]]:=ord(ch)-ord('');
read(ch);
end;
read(ch);m[]:=;
while ch<>' ' do
begin
inc(m[]);m[m[]]:=ord(ch)-ord('');
read(ch);
end;
readln(a,b,c,d);
end;
procedure main;
begin
x[]:=a;x[]:=b;
x:=f(x,m);
y[]:=c;y[]:=d;
y:=x*y;
y:=f(y,n);
y:=y*x;
writeln((y[]+y[]) mod p);
end;
begin
assign(input,'matrix.in');assign(output,'matrix.out');
reset(input);rewrite(output);
init;
main;
close(input);close(output);
end.
无限orz!
看来养成一个好的代码风格是必要的
NOI2013矩阵游戏的更多相关文章
- bzoj 3240: [Noi2013]矩阵游戏 矩阵乘法+十进制快速幂+常数优化
3240: [Noi2013]矩阵游戏 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 613 Solved: 256[Submit][Status] ...
- BZOJ 3240: [Noi2013]矩阵游戏
3240: [Noi2013]矩阵游戏 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 1586 Solved: 698[Submit][Status ...
- BZOJ 3240([Noi2013]矩阵游戏-费马小定理【矩阵推论】-%*s-快速读入)
3240: [Noi2013]矩阵游戏 Time Limit: 10 Sec Memory Limit: 256 MB Submit: 123 Solved: 73 [ Submit][ St ...
- (十进制高速幂+矩阵优化)BZOJ 3240 3240: [Noi2013]矩阵游戏
题目链接: http://www.lydsy.com/JudgeOnline/problem.php?id=3240 3240: [Noi2013]矩阵游戏 Time Limit: 10 Sec M ...
- P1397 [NOI2013]矩阵游戏(递推)
P1397 [NOI2013]矩阵游戏 一波化式子,$f[1][m]=a^{m-1}+b\sum_{i=0}^{m-2}a^i$,用快速幂+逆元求等比数列可以做到$logm$ 设$v=a^{m-1}, ...
- 【bzoj3240】 Noi2013—矩阵游戏
http://www.lydsy.com/JudgeOnline/problem.php?id=3240 (题目链接) 题意$${F[1][1]=1}$$$${F[i][j]=a*F[i][j-1]+ ...
- BZOJ3240 [Noi2013]矩阵游戏
本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000作者博客:http://www.cnblogs.com/ljh2000-jump/转 ...
- 3240: [Noi2013]矩阵游戏
Description 婷婷是个喜欢矩阵的小朋友,有一天她想用电脑生成一个巨大的n行m列的矩阵(你不用担心她如何存储).她生成的这个矩阵满足一个神奇的性质:若用F[i][j]来表示矩阵中第i行第j列的 ...
- 洛谷P1397 [NOI2013]矩阵游戏
矩阵快速幂+费马小定理 矩阵也是可以跑费马小定理的,但是要注意这个: (图是盗来的QAQ) 就是说如果矩阵a[i][i]都是相等的,那么就是mod p 而不是mod p-1了 #include< ...
随机推荐
- Spark Streaming揭秘 Day29 深入理解Spark2.x中的Structured Streaming
Spark Streaming揭秘 Day29 深入理解Spark2.x中的Structured Streaming 在Spark2.x中,Spark Streaming获得了比较全面的升级,称为St ...
- Spark Streaming揭秘 Day26 JobGenerator源码图解
Spark Streaming揭秘 Day26 JobGenerator源码图解 今天主要解析一下JobGenerator,它相当于一个转换器,和机器学习的pipeline比较类似,因为最终运行在Sp ...
- DOM基础总结
一.简介 1.什么是DOM 文档对象模型(Document Object Model,DOM)是一种用于HTML和XML文档的编程接口.它给文档提供了一种结构化的表示方法,可以改变文档的内容和呈现方式 ...
- The Best Rank (25)(排名算法)
To evaluate the performance of our first year CS majored students, we consider their grades of three ...
- EXPLAIN句法 优化表结构
EXPLAIN tbl_name or EXPLAIN SELECT select_options EXPLAIN tbl_name是DESC[RIBE] tbl_name或SHOW COLUMNS ...
- 退出telnet
telnet时,很多时候通过ctrl+c依然无法退出.可以采取下面的方式进行退出: ctrl+],然后进入 telnet>,然后输入q或quit即可.
- ios设备 分辨率(转)
1 iOS设备的分辨率 iOS设备,目前最主要的有3种(Apple TV等不在此讨论),按分辨率分为两类 iPhone/iPod Touch 普屏分辨率 320像素 x 480像素 Retina ...
- First Groovy
class Sample { def names = ["anna", "annie", "tommy", "bobby" ...
- CSS两列及三列自适应布局方法整理
布局 自适应 两列 三列 在传统方法的基础上加入了Flex布局并阐述各方法的优缺点,希望对大家有所帮助.先上目录: 两列布局:左侧定宽,右侧自适应 方法一:利用float和负外边距 方法二:利用外边距 ...
- 1020: [SHOI2008]安全的航线flight - BZOJ
Description在设计航线的时候,安全是一个很重要的问题.首先,最重要的是应采取一切措施确保飞行不会发生任何事故,但同时也需要做好最坏的打算,一旦事故发生,就要确保乘客有尽量高的生还几率.当飞机 ...