【费马小定理+矩阵快速幂】HDU4549——M斐波那契数列

时间:2023-11-25 17:58:20

【题目大意】

M斐波那契数列F[n]是一种整数数列,它的定义如下:
F[0] = a
F[1] = b
F[n] = F[n-1] * F[n-2] ( n > 1 )
现在给出a, b, n,求出F[n]的值。

【思路】

【费马小定理+矩阵快速幂】HDU4549——M斐波那契数列

 #include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
const ll MOD=;
int a,b,n; void mul(ll A[][],ll B[][])
{
ll C[][];
memset(C,,sizeof(C));
for (int i=;i<;i++)
for (int j=;j<;j++)
for (int k=;k<;k++)
C[i][j]=(C[i][j]+A[i][k]*B[k][j])%(MOD-);
for (int i=;i<;i++)
for (int j=;j<;j++) B[i][j]=C[i][j];
} void fibonacci(ll &fa,ll &fb)
{
ll acc[][]={{,},{,}};
ll now[][]={{,},{,}};
int res=n-;
while (res>)
{
if (res&) mul(acc,now);
mul(acc,acc);
res>>=;
}
fa=now[][];
fb=now[][];
} ll get_ans(ll x,ll p)
{
ll now=x;
ll res=p;
ll ret=;
while (res>)
{
if (res&!=) ret=(ret*now)%MOD;
now=(now*now)%MOD;
res>>=;
}
return ret;
} int main()
{
while (~scanf("%d%d%d",&a,&b,&n))
{
if (n==) cout<<a<<endl;
else if (n==) cout<<b<<endl;
else
{
ll fa=,fb=;
fibonacci(fa,fb);
printf("%lld\n",(get_ans(a,fa)*get_ans(b,fb))%MOD);
}
}
return ;
}

【附录:用多维数组名作函数参数

如果用二维数组名作为实参和形参,在对形参数组声明时,必须指定第二维(即列)的大小,且应与实参的第二维的大小相同。第一维的大小可以指定,也可以不指定。如:
  int array[3][10];  //形参数组的两个维都指定
  int array[][10];  //第一维大小省略

二者都合法而且等价。但是不能把第二维的大小省略。下面的形参数组写法不合法:
  int array[][];  //不能确定数组的每一行有多少列元素
  int array[3][];  //不指定列数就无法确定数组的结构
在第二维大小相同的前提下,形参数组的第一维可以与实参数组不同。例如,实参数组定义为:int score[5][10]; 而形参数组可以声明为:
  int array[3][10];  //列数与实参数组相同,行数不同
  int array[8][10];

这时形参二维数组与实参二维数组都是由相同类型和大小的一维数组组成的,实参数组名score代表其首元素(即第一行)的起始地址,系统不检查第一维的大小。
如果是三维或更多维的数组,处理方法是类似的。