Description
求数列 f [ n ] = f [ n − 1 ] + f [ n − 2 ] + 1 f[n]=f[n-1]+f[n-2]+1 f[n]=f[n−1]+f[n−2]+1的第 N N N项. f [ 1 ] = 1 , f [ 2 ] = 1 f[1]=1,f[2]=1 f[1]=1,f[2]=1.
Input
n ( 1 < n < 2 31 − 1 ) n(1<n<2^{31}-1) n(1<n<231−1)
Output
第 N N N项的结果 m o d mod mod 9973 9973 9973
Sample Input
12345
Sample Output
8932
正题:
我们考虑将
[
f
n
−
1
f
n
−
2
1
]
\begin{bmatrix}f_{n-1} & f_{n-2} & 1\end{bmatrix}
[fn−1fn−21]转换为
[
f
n
f
n
−
1
1
]
\begin{bmatrix}f_{n} & f_{n-1 }& 1\end{bmatrix}
[fnfn−11]
因为
f
[
n
]
=
f
[
n
−
1
]
+
f
[
n
−
2
]
+
1
f[n]=f[n-1]+f[n-2]+1
f[n]=f[n−1]+f[n−2]+1,所以我们很快得出转移矩阵:
[
1
1
0
1
0
0
1
0
1
]
\begin{bmatrix} 1 & 1 & 0\\1 & 0 & 0\\1 & 0 & 1\end{bmatrix}
⎣⎡111100001⎦⎤
code:
#include<iostream>
#include<cstdio>
using namespace std;
int n;
int a[10][10];
int ans[10][10];
int p=9973;
void multi()
{
long long c[10][10];
for(long long i=1; i<=3; i++)
for(long long j=1; j<=3; j++)
c[i][j]=0;
for(long long i=1; i<=3; i++)
for(long long j=1; j<=3; j++)
for(long long k=1; k<=3; k++)
c[i][j]=(c[i][j]+ans[i][k]*a[k][j])%p;
for(long long i=1; i<=3; i++)
for(long long j=1; j<=3; j++)
ans[i][j]=c[i][j];
}
void multi1()
{
long long c[10][10];
for(long long i=1; i<=3; i++)
for(long long j=1; j<=3; j++)
c[i][j]=0;
for(long long i=1; i<=3; i++)
for(long long j=1; j<=3; j++)
for(long long k=1; k<=3; k++)
c[i][j]=(c[i][j]+a[i][k]*a[k][j])%p;
for(long long i=1; i<=3; i++)
for(long long j=1; j<=3; j++)
a[i][j]=c[i][j];
}
void ksm(int k)
{
ans[1][1]=ans[2][2]=ans[3][3]=1;
while(k!=0)
{
if(k&1)
multi();
multi1();
k>>=1;
}
}
void multi2()
{
long long c[10][10];
for(long long i=1; i<=1; i++)
for(long long j=1; j<=3; j++)
c[i][j]=0;
for(long long i=1; i<=1; i++)
for(long long j=1; j<=3; j++)
for(long long k=1; k<=3; k++)
c[i][j]=(c[i][j]+a[i][k]*ans[k][j])%p;
for(long long i=1; i<=1; i++)
for(long long j=1; j<=3; j++)
ans[i][j]=c[i][j];
}
int main()
{
scanf("%d", &n);
a[1][1]=a[1][2]=a[2][1]=a[3][1]=a[3][3]=1;
ksm(n-1);
a[1][1]=a[1][2]=a[1][3]=1;
multi2();
printf("%d", ans[1][2]);
return 0;
}