传送门:bzoj2432
良心题,暴力有 70 p t s 70pts 70pts。。。
显然是分别维护斐波那契数列 m o d    k , m o d    p \mod k,\mod p modk,modp意义下的值。
就题面中的例子( m o d    7 \mod 7 mod7)来看(方便观察性质,每出现一个 0 0 0就新开一行):
1 , 1 , 2 , 3 , 5 , 0 1,1,2,3,5,0 1,1,2,3,5,0
5 , 5 , 3 , 0 5,5,3,0 5,5,3,0
3 , 3 , 6 , 2 , 0 3,3,6,2,0 3,3,6,2,0
2 , 2 , 4 , 6 , 3 , 2 , 5 , 0 2,2,4,6,3,2,5,0 2,2,4,6,3,2,5,0
5 , 5 , 3 , 0 5,5,3,0 5,5,3,0
. . . ... ...
观察到一些特殊的性质:
每行开头必然是两个相同的数(设为 s t i st_i sti),且每一行形式相同: s t i ∗ f 1 , s t i ∗ f 2 , s t i ∗ f 3 . . . st_i*f_1,st_i*f_2,st_i*f_3... sti∗f1,sti∗f2,sti∗f3...
如果有循环节的话,循环行数必然不超过 k k k。
循环节中每行结尾都是以 s t i ∗ f l e n ≡ 1 m o d    k st_i*f_{len}\equiv 1\mod k sti∗flen≡1modk结束的:
3 , 3 , 6 , 2 , 0 3,3,6,2,0 3,3,6,2,0
2 , 2 , 4 , 6 , 3 , 2 , 5 , 0 , 5 , 5 , 3 , 0 2,2,4,6,3,2,5,0,5,5,3,0 2,2,4,6,3,2,5,0,5,5,3,0
这样表示比较直观。
只需要找到最小的 f i ≡ s t i − 1 m o d    k f_i\equiv st_i^{-1}\mod k fi≡sti−1modk就得到了第 i i i行的长度,且 s t i + 1 = s t i ∗ f l e n − 1 m o d    k st_{i+1}=st_i*f_{len-1}\mod k sti+1=sti∗flen−1modk
具体来说,用 v s i vs_i vsi表示最小的模 k k k等于 i i i的斐波那契数项。而可以证明模 k k k意义下,斐波那契循环节长度不超过 6 ∗ k 6*k 6∗k,且循环节中前三个数必然为 0 , 1 , 1 0,1,1 0,1,1(一个paper)
e x g c d exgcd exgcd求解逆元,但 k k k不保证是质数,无解的情况就等同于没有特殊的减一操作,直接矩乘。
所以我们只需要在不超过 k k k次内找出循环节然后矩阵快速幂即可。
行内的转移矩阵:
[ 1 1 0 1 0 0 0 0 1 ] \left[ \begin{matrix} 1& 1& 0\\ 1&0&0\\ 0&0&1 \end{matrix} \right] ⎣⎡110100001⎦⎤
行之间的转移矩阵:
[ 1 0 0 0 1 0 − 1 0 1 ] \left[ \begin{matrix} 1& 0& 0\\ 0&1&0\\ -1&0&1 \end{matrix} \right] ⎣⎡10−1010001⎦⎤
初始矩阵:
[ f 0 = 0 f − 1 = 1 1 ] \left[ \begin{matrix} f_0=0&f_{-1}=1 &1 \end{matrix} \right] [f0=0f−1=11]
#include
using namespace std;
typedef long long ll;
const int N=1e6+10;
ll n;bool fwg[N];
int K,mod,pr,vs[N],f[N*6],inv[N];
inline int ad(int x,int y){x+=y;return x>=mod?x-mod:x;}
inline int dc(int x,int y){x-=y;return x<0?x+mod:x;}
struct Mat{
int g[3][3];
inline int *operator[](int x){return g[x];}
Mat operator *(const Mat&ky)const{
Mat re;register int i,j,k,v;
for(i=0;i<3;++i)
for(j=0;j<3;++j){
for(v=k=0;k<3;++k) v=ad(v,(ll)g[i][k]*ky.g[k][j]%mod);
re.g[i][j]=v;
}
return re;
}
inline void itia()
{
register int i,j;
for(i=0;i<3;++i)
for(j=0;j<3;++j)
g[i][j]=(i==j);
}
}A,B,C,D,ori,mt[N];
inline Mat fp(Mat a,ll y)
{
ori.itia();
for(;y;y>>=1,a=a*a)
if(y&1) ori=ori*a;
return ori;
}
int exgcd(int a,int b,ll &x,ll &y)
{
if(!b) {x=1;y=0;return a;}
int re=exgcd(b,a%b,y,x);y-=x*(a/b);
return re;
}
inline int gtiv(int a,int b)
{
ll x,y;
return exgcd(a,b,x,y)==1?((x%b+b)%b):(-1);
}
int main(){
int i,x,z,len;ll cot;
scanf("%lld%d%d",&n,&K,&mod);
f[1]=f[2]=1;
for(i=3;;++i){
f[i]=(f[i-1]+f[i-2])%K;
if(!vs[f[i]]) vs[f[i]]=i;
if(f[i]==1 && f[i]==f[i-1]) break;
}
A[0][0]=A[0][1]=A[1][0]=A[2][2]=1;
B[0][0]=B[1][1]=B[2][2]=1;B[2][0]=mod-1;
C[0][1]=C[0][2]=1;
for(z=1,pr=0;n;){
if(!inv[z]) inv[z]=gtiv(z,K);
if(inv[z]==-1) {C=C*fp(A,n);break;}
if(pr || (!fwg[z])){
len=vs[inv[z]];
if(!len || n