题目链接

  先用后缀数组给串排好序。dc3 O(n)

二分答案+贪心check

  答案的长度len=(n+k-1)/k

如果起点为i长为len串大于当前枚举的答案,i的长度取len-1

  从起点判断k个串的长度是否大于等于n

check的时候最多枚举len个起点,每个位置需要枚举n/len个串,时间复杂度O(n),总的时间复杂度O(nlogn+n)

#include <iostream>
#include <algorithm>
#include <string>
#include <fstream>
using namespace std;
#define ll long long
#define MAXN 1000009
int sa[MAXN], pos[MAXN], Rank[MAXN], r[MAXN];
int d, n, k;
string s;
#define F(x) ((x)/3+((x)%3==1?0:tb))
#define G(x) ((x)<tb?(x)*3+1:((x)-tb)*3+2)
int wa[MAXN], wb[MAXN], wv[MAXN], Ws[MAXN];
inline int c0 (int *r, int a, int b) {
return r[a] == r[b] && r[a + ] == r[b + ] && r[a + ] == r[b + ];
}
int c12 (int k, int *r, int a, int b) {
if (k == ) return r[a] < r[b] || r[a] == r[b] && c12 (, r, a + , b + );
else return r[a] < r[b] || r[a] == r[b] && wv[a + ] < wv[b + ];
}
inline void sort (int *r, int *a, int *b, int n, int m)
{
int i;
for (i = ; i < n; i++) wv[i] = r[a[i]];
for (i = ; i < m; i++) Ws[i] = ;
for (i = ; i < n; i++) Ws[wv[i]]++;
for (i = ; i < m; i++) Ws[i] += Ws[i - ];
for (i = n - ; i >= ; i--) b[--Ws[wv[i]]] = a[i];
return;
}
inline void dc3 (int *r, int *sa, int n, int m)
{
int i, j, *rn = r + n, *san = sa + n, ta = , tb = (n + ) / , tbc = , p;
r[n] = r[n + ] = ;
for (i = ; i < n; i++) if (i % != ) wa[tbc++] = i;
sort (r + , wa, wb, tbc, m);
sort (r + , wb, wa, tbc, m);
sort (r, wa, wb, tbc, m);
for (p = , rn[F (wb[])] = , i = ; i < tbc; i++)
rn[F (wb[i])] = c0 (r, wb[i - ], wb[i]) ? p - : p++;
if (p < tbc) dc3 (rn, san, tbc, p);
else for (i = ; i < tbc; i++) san[rn[i]] = i;
for (i = ; i < tbc; i++) if (san[i] < tb) wb[ta++] = san[i] * ;
if (n % == ) wb[ta++] = n - ;
sort (r, wb, wa, ta, m);
for (i = ; i < tbc; i++) wv[wb[i] = G (san[i])] = i;
for (i = , j = , p = ; i < ta && j < tbc; p++)
sa[p] = c12 (wb[j] % , r, wa[i], wb[j]) ? wa[i++] : wb[j++];
for (; i < ta; p++) sa[p] = wa[i++];
for (; j < tbc; p++) sa[p] = wb[j++];
return;
}
inline bool check (int x) {
for (int i = ; i < d; i++) {
int j = i, t = ;
while (t < k) {
if (Rank[j % n] <= x) j += d;
else
j += d - ;
t++;
}
if (j - i >= n) return ;
}
return ;
}
int main() {
while (scanf ("%d %d", &n, &k) == ) {
cin >> s; s += s;
d = (n + k - ) / k;
for (int i = ; i < (n << ); i++) r[i] = s[i] - '';
r[n << ] = ;
dc3 (r, sa, s.size() + , );
for (int i = , p = ; i <= (n << ); i++)
Rank[sa[i]] = ++p;
for (int i = , p = ; i <= (n << ); i++)
if (sa[i] < n ) pos[++p] = sa[i];
int l = , r = n, last = -;
while (l <= r) {
int mid = (l + r) >> ;
if (check (Rank[pos[mid]]) ) last = pos[mid], r = mid - ;
else
l = mid + ;
}
for (int i = last; i < last + d; i++)
putchar (s[i]);
putchar ();
}
}

LA 6856 Circle of digits 解题报告的更多相关文章

  1. BZOJ 3929 Circle of digits 解题报告

    首先,我们可以得到最高位的位数为:\(\lfloor\frac{n+k-1}{n}\rfloor\),记作 \(E\). 然后给这 \(n\) 个长为 \(E\) 的数字排序,后缀数组 \(O((n+ ...

  2. 【LeetCode】738. Monotone Increasing Digits 解题报告(Python)

    [LeetCode]738. Monotone Increasing Digits 解题报告(Python) 标签(空格分隔): LeetCode 作者: 负雪明烛 id: fuxuemingzhu ...

  3. 【LeetCode】402. Remove K Digits 解题报告(Python)

    [LeetCode]402. Remove K Digits 解题报告(Python) 标签(空格分隔): LeetCode 作者: 负雪明烛 id: fuxuemingzhu 个人博客: http: ...

  4. UVALive - 6856 Circle of digits 后缀数组+二分

    题目链接: http://acm.hust.edu.cn/vjudge/problem/82135 Circle of digits Time Limit: 3000MS 题意 把循环串分割成k块,让 ...

  5. 【LeetCode】 258. Add Digits 解题报告(Java & Python)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 方法一:递归 方法二:减1模9 方法三:直接模9 日 ...

  6. 【LeetCode】949. Largest Time for Given Digits 解题报告(Python)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 日期 题目地址:https://leetcode.c ...

  7. 【LeetCode】357. Count Numbers with Unique Digits 解题报告(Python & C++)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 日期 题目地址:https://leetcode.c ...

  8. LeetCode 788 Rotated Digits 解题报告

    题目要求 X is a good number if after rotating each digit individually by 180 degrees, we get a valid num ...

  9. LeetCode 258 Add Digits 解题报告

    题目要求 Given a non-negative integer num, repeatedly add all its digits until the result has only one d ...

随机推荐

  1. JPush三分钟之后的事 fragmentActivity(一)

    JPush是我用过的平台中最快的,比leanCloud短信验证快的多,当然leanCloud正在飞速的成长 另外江宏 是我的偶像啊:http://tech.sina.com.cn/i/2015-01- ...

  2. Linux学习笔记21——线程同步的两种方式

    一  用信号量同步 1 信号量函数的名字都以sem_开头,线程中使用的基本信号量函数有4个 2 创建信号量 #include<semaphore.h> int sem_init(sem_t ...

  3. scrollView的用法

    - (void)viewDidLoad {     [super viewDidLoad];           scrollView = [[UIScrollView alloc] initWith ...

  4. xrdp远程 & watchdog 启用与测试 & WebRTC

    sudo apt-get install xrdp sudo apt-get install vnc4server tightvncserver echo "xfce4-session&qu ...

  5. hdu 4405概率dp

    #include <cstdio> #include <cstring> #include <iostream> #include <cmath> #i ...

  6. 局域网内linux由ip反解析主机名

  7. HDU 4432 Sum of divisors (进制模拟)

    三个小函数 getdiv();        求因子 getsum();     求平方和 change();     转换成该进制 #include <cstdio> #include ...

  8. Android Studio打包签名全过程

    Step1:Build-----Generate Step2:签名文件保存位置,习惯性的保存在项目的根目录下E:\Users\StudioProject\SmartHust\smarthust.jks ...

  9. head first-----decorate design pattern

    浅谈设计模式之------装饰者模式     首先给出装饰者模式的定义吧:              动态的将责任附加到对象上,若是要扩展功能,装饰者提供了比继承更加具有弹性的替代方案.     其中 ...

  10. 用bootstrap结合php搭建MIS系统框架【转载】

    bootstrap目前比较火,抽空简单了解一下 官方介绍:Bootstrap, from Twitter基于HTML,CSS,JAVASCRIPT的简洁灵活的流行前端框架及交互组件集 http://t ...