[BZOJ]3671 随机数生成器(Noi2014)

  洛谷上卡不过去的朋友们可以来看看小C的程序(小C才不是标题党呢!)

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,vi,表示第 i 次额外交换操作将交换 T(ui )和 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

  

Solution

  奇技淫巧题,卡时卡空间,考出了WC的画风。

  相信O(nmlog(n+m))的做法大家都会。

  一通模拟过后,基本思路就是贪心,每次选最小的数,能加就加,加入之后更新限制范围……你懂的。

  

  很多做法,一种是用set或平衡树来维护限制范围,还有一种就是直接在数组上二分,插入时暴力插入。

  小C觉得第一种常数很大并且懒得写平衡树,所以就写了第二种。复杂度O(nmlog(n+m)+(n+m)^2)。

  注意代码上各种会影响常数的细节,如取模等。

  当你绞尽脑汁试图扭曲你的代码去卡过那万恶的常数时,不如再想想强有力的剪枝。

  我们发现每条斜线上有且仅有一个数会被记入答案,不妨开一个bool数组来标记一下。

  整整快了一倍有木有!

#include <cstdio>
#include <algorithm>
#define MM 25000005
#define MN 5005
#define make(a1,a2) ((pr){a1,a2})
#define l(a) (a.x)
#define r(a) (a.y)
using namespace std;
struct pr{int x,y;}pa[MN<<1];
int X,A,B,C,D;
int n,m,gs,p,nm,a[MM],b[MM];
bool u[MN<<1];

inline int read()
{
    int n=0,f=1; char c=getchar();
    while (c<‘0‘ || c>‘9‘) {if(c==‘-‘)f=-1; c=getchar();}
    while (c>=‘0‘ && c<=‘9‘) {n=n*10+c-‘0‘; c=getchar();}
    return n*f;
}

bool cmp(pr a,pr b) {return a.x<b.x || a.x==b.x && a.y<b.y;}

int main()
{
    register int i,j,x;
    pr z;
    X=read(); A=read(); B=read(); C=read(); D=read();
    n=read(); m=read(); p=read(); nm=n*m;
    for (i=1;i<=nm;++i)
    {
        X=((1LL*A*X+B)*X+C)%D;
        a[i]=i; swap(a[i],a[(X%i)+1]);
    }
    while (p--) swap(a[read()],a[read()]);
    for (i=1;i<=nm;++i) b[a[i]]=i-1;
    pa[gs=1]=make(MN,MN);
    for (i=1;i<=nm;++i)
    {
        z=make(b[i]/m,b[i]%m);
        if (u[l(z)+r(z)]) continue;
        x=lower_bound(pa+1,pa+gs+1,z,cmp)-pa;
        if (r(z)>r(pa[x])||r(z)<r(pa[x-1])) continue;
        for (j=gs++;j>=x;--j) pa[j+1]=pa[j];
        u[l(z)+r(z)]=true; pa[x]=z;
        if (i>1) putchar(‘ ‘); printf("%d",i);
    }
}

Last Word

  在考场上谁会知道自己的程序会被卡成什么样呢?大概只有写完程序发现跑了5.X秒然后在程序末尾加上"orz ditoly"来保平安了吧。

  不过仔细想想这样的优化还是能想得到的。

  看看BZOJ上一个个只跑了20s的dalao们,感觉没有人A还要求放宽时限的洛谷还是Too Young。

  n+e光速读入用在这道题上是杯水车薪。

时间: 2024-12-14 01:43:10

[BZOJ]3671 随机数生成器(Noi2014)的相关文章

BZOJ 3122 随机数生成器

http://www.lydsy.com/JudgeOnline/problem.php?id=3122 题意:给出p,a,b,x1,t 已知xn=a*xn-1+b%p,求最小的n令xn=t 首先,若x1=t,则返回1 若a=0,则若b=t 返回2,否则无解 若a=1,则T=t-x1+p%p,可以列出方程 b*x+p*y==T % p 若a>=2,则根据等比数列和可得 xn=t=x1*a^(n-1)+b*(a^(n-1)-1)/(a-1) %p 由于p为质数,所以令c=inv[a-1]=(a-1

【BZOJ】【3671】【NOI2014】随机数生成器

贪心 嗯……其实生成这个矩阵就是一个$O(n^2)$的模拟 = = 然后?字典序最小?贪心呗= =能选1就选1,然后能选2就选2…… 我们发现,对于矩阵(1,1)~(n,m),假设1的位置是(x,y),那么我们选完1以后,可选的范围变成了:(1,1)~(x,y) & (x,y)~(n,m),也就是将一个矩阵拆成四块,我们可以在左上和右下两块中递归地进行选择…… 那么我们每次选完之后,新的可选的范围其实暴力O(n)维护就可以了,因为我们总共只选$O(n)$次,每次维护的复杂度是$O(n)$,总复杂

NOI2014 随机数生成器.

3757. [NOI2014]随机数生成器 (Standard IO) Time Limits: 5000 ms  Memory Limits: 262144 KB Description Input 输入文件的第 1 行包含 5 个整数,依次为 x0, a, b, c, d,描述小 H 采用的随机数生成算法所需的随机种子. 第 2 行包含三个整数 N, M, Q,表示小 H 希望生成一个 1 到 N × M 的排列来填入她 N 行 M 列的棋盘,并且小 H 在初始的 N × M 次交换操作后,

【BZOJ3671】[Noi2014]随机数生成器 暴力

[BZOJ3535][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 ) 的值.

【BZOJ 3122】 [Sdoi2013]随机数生成器 (BSGS)

3122: [Sdoi2013]随机数生成器 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 1442  Solved: 552 Description Input 输入含有多组数据,第一行一个正整数T,表示这个测试点内的数据组数. 接下来T行,每行有五个整数p,a,b,X1,t,表示一组数据.保证X1和t都是合法的页码. 注意:P一定为质数 Output 共T行,每行一个整数表示他最早读到第t页是哪一天.如果他永远不会读到第t页,输出-1. Sa

2875: [Noi2012]随机数生成器 - BZOJ

DescriptionInput 包含6个用空格分割的m,a,c,X0,n和g,其中a,c,X0是非负整数,m,n,g是正整数. Output 输出一个数,即Xn mod gSample Input 11 8 7 1 5 3 Sample Output2 快速幂+快速乘 1 type 2 matrix=array[1..2,1..2]of int64; 3 var 4 a,c,p,x0,n,g:int64; 5 x,y:matrix; 6 7 function kc(x,y:int64):int

BZOJ 2875: [Noi2012]随机数生成器( 矩阵快速幂 )

矩阵快速幂...+快速乘就OK了 -------------------------------------------------------------------------------------- #include<bits/stdc++.h> using namespace std; typedef long long ll; ll MOD, a, c, x, n, g; ll MUL(ll a, ll b) { ll ans = 0; for(; b; b >>= 1

【矩阵乘】【NOI 2012】【cogs963】随机数生成器

963. [NOI2012] 随机数生成器 ★★ 输入文件:randoma.in 输出文件:randoma.out 简单对照 时间限制:1 s 内存限制:128 MB **[问题描写叙述] 栋栋近期迷上了随机算法,而随机数是生成随机算法的基础.栋栋准备使用线性同余法(Linear Congruential Method)来生成一个随机数列.这样的方法须要设置四个非负整数參数m,a,c,X[0],依照以下的公式生成出一系列随机数{Xn}: X[n+1]=(aX[n]+c) mod m 当中mod

C语言下泊松分布以及指数分布随机数生成器实现

最近实验室的项目需要实现模拟文件访问序列,要求单位时间内的数据请求次数符合泊松分布,而两次请求见的时间间隔符合指数分布.没办法只好重新捡起已经丢掉多时的概率知识.于是也就有了这篇关于在C语言下符合泊松分布和指数分布的随机数生成器的实现. 泊松分布 在实际的事例中,当某一事件,比如进站乘客数量,电话交换机接收到的通话请求以固定的瞬时速率λ独立且随机地出现时,就可以认为该事件在单位时间内发生的次数符合泊松分布. 首先必须由二项分布引出: 如果做一件事情成功的概率是 p 的话,那么独立尝试做这件事情