[bzoj3671][Noi2014][随机数生成器] (贪心+位运算+卡空间)

Description

Input

第1行包含5个整数,依次为 x_0,a,b,c,d ,描述小H采用的随机数生成算法所需的随机种子。第2行包含三个整数 N,M,Q ,表示小H希望生成一个1到 N×M 的排列来填入她 N 行 M 列的棋盘,并且小H在初始的 N×M 次交换操作后,又进行了 Q 次额外的交换操作。接下来 Q 行,第 i 行包含两个整数 u_i,v_i,表示第 i 次额外交换操作将交换 T_(u_i )和 T_(v_i ) 的值。

Output

输出一行,包含 N+M-1 个由空格隔开的正整数,表示可以得到的字典序最小的路径序列。

 

Sample Input

1 3 5 1 71 
3 4 3 
1 7 
9 9 
4 9

Sample Output

1 2 6 8 9 12

HINT

 


本题的空间限制是 256 MB,请务必保证提交的代码运行时所使用的总内存空间不超过此限制。


一个32位整数(例如C/C++中的int和Pascal中的Longint)为4字节,因而如果在程序中声明一个长度为 1024×1024 的32位整型变量的数组,将会占用 4 MB 的内存空间。





2≤N,M≤5000


0≤Q≤50000


0≤a≤300


0≤b,c≤108


0≤x0<d≤1081≤ui,vi≤N×M

 

Solution

字典序最小。。。考虑贪心咯

从小到大选数,可以就选

因为方阵中的路径是一段连续区间,我们对每次选完后剩余的数做限制,标记一下就行

#include <stdio.h>
#include <memory.h>
#define MaxN 5003
#define MaxBuf 1<<20
#define L long long
#define RG register
#define dmin(a,b) ((a)<(b)?(a):(b))
#define dmax(a,b) ((a)>(b)?(a):(b))
char B[MaxBuf],*p=B;
template<class Type>inline void Rin(RG Type &x){
    x=0;
    while(*p<'0'||*p>'9')p++;
    while(*p>='0'&&*p<='9')
        x=x*10+*p++-'0';
}
template<class Type>inline void exc(RG Type &x,RG Type &y){
    RG Type z=x; x=y; y=z;
}
L x_0,a,b,c,d;
int n,m,_q,lef[MaxN],rig[MaxN],g[MaxN][MaxN],t[MaxN*MaxN];
inline int ram(){
    return x_0=(a*x_0*x_0%d+b*x_0%d+c)%d;
}
inline void limina(RG int x,RG int y){
    for(RG int i=1;i<x;i++)
        rig[i]=dmin(rig[i],y);
    for(RG int i=x+1;i<=n;i++)
        lef[i]=dmax(lef[i],y);
}
int main(){
    memset(rig,0x3f,sizeof rig);
    fread(p,1,MaxBuf,stdin);
    Rin(x_0),Rin(a),Rin(b),Rin(c),Rin(d),Rin(n),Rin(m),Rin(_q);
    for(RG int i=1;i<=n*m;i++){
        t[i]=i; exc(t[i],t[ram()%i+1]);
    }
    while(_q--){
        RG int x,y;
        Rin(x),Rin(y);
        exc(t[x],t[y]);
    }
    for(RG int i=1;i<=n;i++)
        for(RG int j=1;j<=m;j++){
            static size_t tim;
            g[i][j]=t[++tim];
        }
    for(RG int i=1;i<=n;i++)
        for(RG int j=1;j<=m;j++)
            t[g[i][j]]=i<<16|j;
    for(RG int i=1;i<=n*m;i++){
        static bool adv;
        RG int x=t[i]>>16,y=t[i]&65535;
        if(lef[x]<=y&&y<=rig[x]){
            if(adv)
                putchar(' ');
            adv=true;
            printf("%d",g[x][y]);
            limina(x,y);
        }
    }
    putchar('\n');
    return 0;
}

 

posted @ 2017-01-26 23:04  keshuqi  阅读(315)  评论(0编辑  收藏  举报