链接:https://www.nowcoder.com/acm/contest/2/C
来源:牛客网

题目描述

shy有一个队列a[1], a[2],…,a[n]。现在我们不停地把头上的元素放到尾巴上。在这过程中我们会得到n个不同的队列,每个队列都是a[k],a[k+1],…,a[n],a[1],…,a[k-1]的形式。在这些队列中,我们可以找到字典序最小的。
shy无聊的时候会给队列的每个元素加一玩。但是为了使得游戏不这么无聊,shy加一以后会给每个元素模m,这样子字典序最小的序列就会变了,生活就变得有趣。
很显然这样子加m次以后,序列会变成原来的样子。所以现在shy想知道,在他没有加一前,加一时,加二时,….,加m-1时字典序最小的序列的第k(和上面的k没有关系)个元素分别是几。

输入描述:

第一行三个整数n,m,k表示序列长度,取模的数和要求的序列的第几个元素。
接下来一行n个整数表示初始序列。

输出描述:

m个整数表示答案。

输入例子:
5 6 3
1 2 1 2 3
输出例子:
1
2
3
5
5
0

-->

示例1

输入

5 6 3
1 2 1 2 3

输出

1
2
3
5
5
0

备注:

对于30%的数据,1≤n,m≤100;
对于100%的数据,1≤n,m≤50000, 1≤k≤n, 0≤a[i]<m;

题解

字符串$hash$。

每个数字都会变且仅变一次零。

如果上一次到这一次没有零产生,那么答案的位置不会改变。

否则重新寻找答案产生的位置。

两个字符串比较字典序,字符串$hash$一下,找$lcp$即可。

#include<bits/stdc++.h>
using namespace std; long long mod[2];
long long base[2];
const int maxn = 1e5 + 10;
int a[maxn], n, m, k;
long long h[2][maxn];
long long b[2][maxn];
int ans[maxn];
vector<int> g[maxn]; void init() {
mod[0] = 1e9 + 7;
mod[1] = 1e9 + 7;
base[0] = 131LL;
base[1] = 313LL;
b[0][0] = b[1][0] = 1LL;
for(int i = 1; i < maxn; i ++) {
for(int t = 0; t < 2; t ++) {
b[t][i] = b[t][i - 1] * base[t] % mod[t];
}
}
} int check(int x, int y, int len) {
if(len == 0) return 1;
if(x > y) swap(x, y);
for(int t = 0; t < 2; t ++) {
long long AA = 0, BB = 0;
if(x > 0) AA = h[t][x - 1] * b[t][len] % mod[t];
if(y > 0) BB = h[t][y - 1] * b[t][len] % mod[t];
long long A = (h[t][x + len - 1] - AA + mod[t]) % mod[t];
long long B = (h[t][y + len - 1] - BB + mod[t]) % mod[t];
if(A != B) return 0;
}
return 1;
} int ok(int x, int y, int f) {
int L = 0, R = n, pos = -1;
while(L <= R) {
int mid = (L + R) / 2;
if(check(x, y, mid)) pos = mid, L = mid + 1;
else R = mid - 1;
}
if(pos == n) return 0;
if((a[x + pos] + f) % m
< (a[y + pos] + f) % m) return 1;
return 0;
} int main() {
init();
scanf("%d%d%d", &n, &m, &k);
k --;
long long A[2];
A[0] = A[1] = 0;
for(int i = 0; i < n; i ++) {
scanf("%d", &a[i]);
a[i + n] = a[i];
g[(m - a[i]) % m].push_back(i);
}
for(int i = 0; i < 2 * n; i ++) {
for(int t = 0; t < 2; t ++) {
A[t] = A[t] * base[t] % mod[t];
A[t] = (A[t] + a[i]) % mod[t];
h[t][i] = A[t];
}
} int p = 0;
for(int i = 1; i < n; i ++) {
if(ok(i, p, 0)) p = i;
}
ans[0] = a[p + k];
for(int i = 1; i < m; i ++) {
if(g[i].size() == 0) {
ans[i] = ans[i - 1] + 1;
} else {
p = g[i][0];
for(int j = 1; j < g[i].size(); j ++) {
if(ok(g[i][j], p, i)) p = g[i][j];
}
ans[i] = (a[p + k] + i) % m;
}
}
for(int i = 0; i < m; i ++) {
printf("%d\n", ans[i]);
}
return 0;
}

牛客练习赛1 C - 圈圈的更多相关文章

  1. 【并查集缩点+tarjan无向图求桥】Where are you @牛客练习赛32 D

    目录 [并查集缩点+tarjan无向图求桥]Where are you @牛客练习赛32 D PROBLEM SOLUTION CODE [并查集缩点+tarjan无向图求桥]Where are yo ...

  2. 牛客练习赛31 B 赞迪卡之声妮莎与奥札奇 逻辑,博弈 B

    牛客练习赛31 B 赞迪卡之声妮莎与奥札奇 https://ac.nowcoder.com/acm/contest/218/B 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 2621 ...

  3. 牛客练习赛31 D 神器大师泰兹瑞与威穆 STL,模拟 A

    牛客练习赛31 D 神器大师泰兹瑞与威穆 https://ac.nowcoder.com/acm/contest/218/D 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 26214 ...

  4. 最小生成树--牛客练习赛43-C

    牛客练习赛43-C 链接: https://ac.nowcoder.com/acm/contest/548/C 来源:牛客网 题目描述 ​ 立华奏是一个刚刚开始学习 OI 的萌新. 最近,实力强大的 ...

  5. 牛客练习赛28-B(线段树,区间更新)

    牛客练习赛28 - B 传送门 题目 qn姐姐最好了~ ​ qn姐姐给你了一个长度为n的序列还有m次操作让你玩, ​ 1 l r 询问区间[l,r]内的元素和 ​ 2 l r 询问区间[l,r]内的 ...

  6. 牛客练习赛26:D-xor序列(线性基)

    链接:牛客练习赛26:D-xor序列(线性基) 题意:小a有n个数,他提出了一个很有意思的问题:他想知道对于任意的x, y,能否将x与这n个数中的任意多个数异或任意多次后变为y 题解:线性基 #inc ...

  7. [堆+贪心]牛客练习赛40-B

    传送门:牛客练习赛40 题面: 小A手头有 n 份任务,他可以以任意顺序完成这些任务,只有完成当前的任务后,他才能做下一个任务 第 i 个任务需要花费  x_i 的时间,同时完成第 i 个任务的时间不 ...

  8. 牛客练习赛 29 E 位运算?位运算!(线段树)

    题目链接  牛客练习赛29E 对$20$位分别建立线段树.首先$1$和$2$可以合起来搞(左移右移其实是等效的) 用个lazy标记下.转移的时候加个中间变量. $3$和$4$其实就是区间$01$覆盖操 ...

  9. 牛客练习赛22C Bitset

    牛客练习赛22C 一共有 n个数,第 i 个数是 xi  xi 可以取 [li , ri] 中任意的一个值. 设 ,求 S 种类数. 感觉二进制真是一个神奇的东西. #include <iost ...

随机推荐

  1. [php]php时间格式化

    1.将毫秒转化为时间格式 date("Y-m-d H:i:s",$millsec);

  2. 用Emacs看电影

    大多数人用emacs听歌,我却喜欢用emacs看电影.用 EMMS 和 mplayer 结合,看电影真是太方便了. 不要从源里安装EMMS,它可能给你安装别的播放器,没必要,我们有 mplayer 足 ...

  3. 【转】C# Graphics类详解

    Brush 类 .NET Framework 4 定义用于填充图形形状(如矩形.椭圆.饼形.多边形和封闭路径)的内部的对象. 属于命名空间:  System.Drawing 这是一个抽象基类,不能进行 ...

  4. Python练习-函数(方法)的定义和应用

    需求:对文件进行增删改查,使用函数调用的方式完成操作 # 编辑者:闫龙 import MyFuncation; Menu = ["查询","添加"," ...

  5. jquery $.post() 向php传值 实现简单的二级联动

    更多内容推荐微信公众号,欢迎关注: 1 其中selectid是一个下拉菜单的id $().ready(function () { $("#selectid").change(fun ...

  6. OpenCV 用二进制位表示 type & channels 的方式

    OpenCV 的类型与通道的表示方法. 参考文件 https://github.com/opencv/opencv/blob/05b15943d6a42c99e5f921b7dbaa8323f3c04 ...

  7. sru源码--language model

    import sys import os import argparse import time import random import math import numpy as np import ...

  8. oracle11g的冷热备份

    1.冷备份 如果数据库可以正常关闭,而且允许关闭足够长的时间,那么就可以采用冷备份(脱机备份),可以是归档冷备份,也可以是非归档冷备份.其方法是首先关闭数据库,然后备份所有的物理文件,包括数据文件.控 ...

  9. 通过`__slots__` 节省RAM

    标签(空格分隔): Python进阶 python中,由于创建每个实例都会有成员,这些成员都会被保存在dict中,但是Python不能静态分配RAM,当创建实例时,因此dict的大小会比所需要的内存大 ...

  10. 读书笔记 effective c++ Item 28 不要返回指向对象内部数据(internals)的句柄(handles)

    假设你正在操作一个Rectangle类.每个矩形可以通过左上角的点和右下角的点来表示.为了保证一个Rectangle对象尽可能小,你可能决定不把定义矩形范围的点存储在Rectangle类中,而是把它放 ...