BZOJ1409 : Password

$f[n]\bmod q=p^{Fib[n]}\bmod q=p^{Fib[n]\bmod\varphi(q)}\bmod q$

首先线性筛预处理出所有素数,然后对于每次询问,求出$\varphi(q)$,再用矩阵快速幂求出Fib[n],最后用快速幂求答案即可。

 

#include<cstdio>

typedef long long ll;

const int N=46341;

int T,i,j,p[N],tot,vis[N],n,q;ll a,P;

struct mat{

  ll a[2][2];

  inline mat(){a[0][0]=a[0][1]=a[1][0]=a[1][1]=0;}

  inline mat operator*(mat b){

    mat c;

    for(int i=0,j,k;i<2;i++)for(j=0;j<2;j++)for(k=0;k<2;k++)(c.a[i][j]+=a[i][k]*b.a[k][j]%P)%=P;

    return c;

  }

}A,B,C;

inline int phi(int n){

  int t=1,i;

  for(i=1;p[i]*p[i]<=n&&i<=tot;i++)if(n%p[i]==0){n/=p[i],t*=p[i]-1;while(n%p[i]==0)n/=p[i],t*=p[i];}

  if(n>1)t*=n-1;

  return t;

}

inline int fib(int x){

  P=phi(q);

  for(A=B=C=mat(),A.a[0][1]=A.a[1][0]=A.a[1][1]=B.a[1][0]=C.a[0][0]=C.a[1][1]=1;x;x>>=1,A=A*A)if(x&1)C=C*A;

  C=C*B;

  return C.a[0][0];

}

inline int pow(ll a,int b){ll t=1;for(;b;b>>=1,a=a*a%q)if(b&1)t=t*a%q;return t;}

int main(){

  for(i=2;i<N;i++){

    if(!vis[i])p[++tot]=i;

    for(j=1;j<=tot;j++){

      if(i*p[j]>=N)break;

      vis[i*p[j]]=1;

      if(i%p[j]==0)break;

    }

  }

  scanf("%d%lld",&T,&a);

  while(T--){

    scanf("%d%d",&n,&q);

    if(q==1){puts("0");continue;}

    printf("%d\n",pow(a,fib(n)));

  }

  return 0;

}

  

 

你可能感兴趣的:(password)