BZOJ 3671 [Noi 2014] 贪心 解题报告
程序员文章站
2023-12-26 10:25:45
...
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
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define M 5010
int T[M*M],map[M][M],l[M],r[M];
long long seed,a,b,c,d;
int n,m,q;
inline int Rand()
{
return seed=(a*seed*seed%d+b*seed%d+c)%d;
}
void kill(int x,int y)
{
for(int i=1;i<x;i++) r[i]=min(r[i],y);
for(int i=x+1;i<=m;i++) l[i]=max(l[i],y);
}
int main()
{
int flag=0;
scanf("%lld%lld%lld%lld%lld",&seed,&a,&b,&c,&d);
scanf("%d%d%d",&m,&n,&q);
for(int i=1;i<=m;i++) r[i]=n;
for(int i=1;i<=m*n;i++)
{
T[i]=i;
swap(T[i],T[Rand()%i+1]);
}
for(int i=1;i<=q;i++)
{
int x,y;
scanf("%d%d",&x,&y);
swap(T[x],T[y]);
}
int k=0;
for(int i=1;i<=m;i++)
for(int j=1;j<=n;j++)
map[i][j]=T[++k];
for(int i=1;i<=m;i++)
for(int j=1;j<=n;j++)
T[map[i][j]]=i<<16|j;
for(int i=1;i<=m*n;i++)
{
int x=T[i]>>16;
int y=T[i]&65535;
if(y>=l[x]&&y<=r[x])
{
if(flag)
putchar(' ');
flag=1;
printf("%d",map[x][y]);
kill(x,y);
}
}
return 0;
}