BZOJ4471 : 隨機數生成器Ⅱ

Claris發表於2016-04-04

\[\begin{eqnarray*}
x_i&=&x_{i-1}+x_{i-2}\\
x_i^2&=&x_{i-2}^2+x_{i-1}^2+2x_{i-2}x_{i-1}\\
x_{i-1}x_i&=&x_{i-1}^2+x_{i-2}x_{i-1}
\end{eqnarray*}\]

故可以構造轉移矩陣$A$進行遞推。

不妨設$n\geq m$,則可以預處理出$A^0,A^1,...,A^n$以及$A^n,A^{2n},...,A^{nn}$。

那麼查詢某個數的複雜度為$4^2$。

總時間複雜度為$O(4^3n)$。

 

#include<cstdio>
#include<algorithm>
#include<tr1/unordered_map>
#define rep(i) for(int i=0;i<4;i++)
using namespace std;
using namespace std::tr1;
typedef long long ll;
const int N=4,M=100005;
int n,m,lim,q,P,C1,C2,i,j,x,y,d,r,o;
int B[N],A[M][N][N],pA[M][N][N];
unordered_map<ll,int>T;
inline void read(ll&a){char c;while(!(((c=getchar())>='0')&&(c<='9')));a=c-'0';while(((c=getchar())>='0')&&(c<='9'))(a*=10)+=c-'0';}
inline void mul(int a[][N],int b[][N],int c[][N]){
  rep(k)rep(j)if(b[k][j])rep(i)if(a[i][k])c[i][j]=(1LL*a[i][k]*b[k][j]+c[i][j])%P;
}
inline int get(ll t){
  int x=0,*a=pA[t/lim][3],(*b)[N]=A[t%lim];
  rep(i){
    int y=0;
    rep(j)y=(1LL*a[j]*b[j][i]+y)%P;
    x=(1LL*y*B[i]+x)%P;
  }
  return x;
}
inline int ask(int x,int y){
  if(x>n||y>m)return P;
  ll t=1LL*(x-1)*m+y;
  if(T.find(t)!=T.end())return T[t];
  return get(t);
}
void write(int x){
  if(x>=10)write(x/10);
  putchar(x%10+'0');
}
int main(){
  rep(i)A[0][i][i]=pA[0][i][i]=1;
  A[1][0][1]=1;
  A[1][1][0]=A[1][1][1]=1,A[1][1][2]=2;
  A[1][2][1]=A[1][2][2]=1;
  A[1][3][1]=A[1][3][3]=1;
  scanf("%d%d%d%d%d%d",&n,&m,&q,&P,&C1,&C2);
  B[0]=B[3]=1LL*C1*C1%P;
  B[1]=1LL*C2*C2%P;
  B[2]=1LL*C1*C2%P;
  lim=n>m?n:m;
  for(i=2;i<=lim;i++)mul(A[i-1],A[1],A[i]);
  for(i=1;i<=lim;i++)mul(pA[i-1],A[lim],pA[i]);
  for(i=1;i<=q;i++){
    ll x,y;
    read(x),read(y);
    if(T.find(x)==T.end())T[x]=get(x);
    if(T.find(y)==T.end())T[y]=get(y);
    swap(T[x],T[y]);
  }
  for(i=x=y=1,o=ask(1,1);i<n+m-1;i++){
    write(o),putchar(' ');
    d=ask(x+1,y),r=ask(x,y+1);
    if(d<=r)x++,o=d;else y++,o=r;
  }
  return write(o),0;
}

  

相關文章