---恢复内容开始---

题目链接:BZOJ - 1692

题目分析

首先,有个比较简单的贪心思路:如果当前剩余字符串的两端字母不同,就选取小的字母,这样显然是正确的。

然而若两端字母相同,我们怎么选取呢?

这时我们要从两端分别向内部比较,看那一端向内的字符串字典序小。

比如这个字符串 ABCDBA,从左端向内是 ABC.. 从右端向内是 ABD... 所以就选取左端的字符。

这样直接比较是 O(n^2) 的,我们可以使用后缀数组的 Rank 数组来比较。

我们在字符串后加上分隔符,然后再将字符串反转接在后面,求后缀数组的 Rank 数组。

这样就可以快速比较一个前缀,一个后缀的字典序大小了,具体见代码。

比如 ABCDBA ,就存成 ABCDBA#ABDCBA 。

代码

  1. #include <iostream>
  2. #include <cstdio>
  3. #include <cstdlib>
  4. #include <cstring>
  5. #include <cmath>
  6. #include <algorithm>
  7.  
  8. using namespace std;
  9.  
  10. const int MaxL = 60000 + 15;
  11.  
  12. int n;
  13. int A[MaxL], Rank[MaxL], SA[MaxL];
  14. int VA[MaxL], VB[MaxL], VC[MaxL], Sum[MaxL];
  15.  
  16. char S[MaxL], Sout[MaxL];
  17.  
  18. inline bool Cmp(int *a, int x, int y, int l) {
  19. return (a[x] == a[y]) && (a[x + l] == a[y + l]);
  20. }
  21.  
  22. void DA(int *A, int n, int m) {
  23. int *x, *y, *t;
  24. x = VA; y = VB;
  25. for (int i = 1; i <= m; ++i) Sum[i] = 0;
  26. for (int i = 1; i <= n; ++i) ++Sum[x[i] = A[i]];
  27. for (int i = 2; i <= m; ++i) Sum[i] += Sum[i - 1];
  28. for (int i = n; i >= 1; --i) SA[Sum[x[i]]--] = i;
  29. int p, q;
  30. p = 0;
  31. for (int j = 1; p < n; j <<= 1, m = p) {
  32. q = 0;
  33. for (int i = n - j + 1; i <= n; ++i) y[++q] = i;
  34. for (int i = 1; i <= n; ++i) {
  35. if (SA[i] <= j) continue;
  36. y[++q] = SA[i] - j;
  37. }
  38. for (int i = 1; i <= n; ++i) VC[i] = x[y[i]];
  39. for (int i = 1; i <= m; ++i) Sum[i] = 0;
  40. for (int i = 1; i <= n; ++i) ++Sum[VC[i]];
  41. for (int i = 2; i <= m; ++i) Sum[i] += Sum[i - 1];
  42. for (int i = n; i >= 1; --i) SA[Sum[VC[i]]--] = y[i];
  43. t = x; x = y; y = t;
  44. x[SA[1]] = 1; p = 1;
  45. for (int i = 2; i <= n; ++i)
  46. x[SA[i]] = Cmp(y, SA[i], SA[i - 1], j) ? p : ++p;
  47. }
  48. for (int i = 1; i <= n; ++i) Rank[SA[i]] = i;
  49. }
  50.  
  51. int main()
  52. {
  53. scanf("%d", &n);
  54. for (int i = 1; i <= n; ++i) {
  55. cin >> S[i];
  56. A[i] = S[i] - 'A' + 1;
  57. A[2 * n - i + 2] = A[i];
  58. }
  59. A[n + 1] = 27;
  60. A[n * 2 + 2] = 28;
  61. DA(A, n * 2 + 2, 28);
  62. int l = 1, r = n, Top = 0;
  63. while (l <= r) {
  64. if (S[l] != S[r]) {
  65. if (S[l] < S[r]) {
  66. Sout[++Top] = S[l];
  67. ++l;
  68. }
  69. else {
  70. Sout[++Top] = S[r];
  71. --r;
  72. }
  73. continue;
  74. }
  75. if (l == r) {
  76. Sout[++Top] = S[l];
  77. break;
  78. }
  79. if (Rank[l] < Rank[n * 2 - r + 2]) {
  80. Sout[++Top] = S[l];
  81. ++l;
  82. }
  83. else {
  84. Sout[++Top] = S[r];
  85. --r;
  86. }
  87. }
  88. for (int i = 1; i <= Top; ++i) {
  89. printf("%c", Sout[i]);
  90. if (i % 80 == 0) printf("\n");
  91. }
  92. return 0;
  93. }

  

---恢复内容结束---

[BZOJ 1692] [Usaco2007 Dec] 队列变换 【后缀数组 + 贪心】的更多相关文章

  1. BZOJ 1692: [Usaco2007 Dec]队列变换 [后缀数组 贪心]

    1692: [Usaco2007 Dec]队列变换 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 1383  Solved: 582[Submit][St ...

  2. BZOJ 1692: [Usaco2007 Dec]队列变换 (后缀数组/二分+Hash)

    跟BZOJ 4278: [ONTAK2015]Tasowanie一模一样 SA的做法就是把原串倒过来接在原串后面,O(nlogn)O(nlogn)O(nlogn)做后缀数组,就能O(1)O(1)O(1 ...

  3. 1692: [Usaco2007 Dec]队列变换|后缀数组|贪心

    将字符串翻转后接到原串的后面,中间加一个分隔符,每次都贪心选择rankrank小的那个 事实上就是练习一发后缀数组的模板 #include<algorithm> #include<i ...

  4. 【BZOJ1692】[Usaco2007 Dec]队列变换 后缀数组+贪心

    [BZOJ1692][Usaco2007 Dec]队列变换 Description FJ打算带他的N(1 <= N <= 30,000)头奶牛去参加一年一度的“全美农场主大奖赛”.在这场比 ...

  5. BZOJ 1692: [Usaco2007 Dec]队列变换( 贪心 )

    数据 n <= 30000 , 然后 O( n² ) 的贪心也过了..... USACO 数据是有多弱啊 = = ( ps : BZOJ 1640 和此题一模一样 , 双倍经验 ) ------ ...

  6. ●BZOJ 1692 [Usaco2007 Dec]队列变换

    题链: http://www.lydsy.com/JudgeOnline/problem.php?id=1692 题解: 后缀数组,贪心由于每次可以取出旧队列的首部或尾部放在新队列的尾部.所以就需要比 ...

  7. BZOJ 1692: [Usaco2007 Dec]队列变换

    Description FJ打算带他的N(1 <= N <= 30,000)头奶牛去参加一年一度的"全美农场主大奖赛".在这场比赛中,每个参赛者都必须让他的奶牛排成一列 ...

  8. bzoj:1692 [Usaco2007 Dec]队列变换&&1640 [Usaco2007 Nov]Best Cow Line 队列变换

    Description FJ打算带他的N(1 <= N <= 30,000)头奶牛去参加一年一度的“全美农场主大奖赛”.在这场比赛中,每个参赛者都必须让他的奶牛排成一列,然后领她们从裁判席 ...

  9. bzoj 1692: [Usaco2007 Dec]队列变换 ——二分+hash

    Description FJ打算带他的N(1 <= N <= 30,000)头奶牛去参加一年一度的“全美农场主大奖赛”.在这场比赛中,每个参赛者都必须让他的奶牛排成一列,然后领她们从裁判席 ...

随机推荐

  1. richTextBox1 转到行号

      private void button2_Click(object sender, EventArgs e) {     Win32CommonDialog.frm_GOTO frm = new  ...

  2. [RxJS] map vs flatMap

    What's the difference between map and flatmap? First, let's show what map is. To show that, I need a ...

  3. linux lsof命令详解

    linux lsof命令详解 简介 lsof(list open files)是一个列出当前系统打开文件的工具.在linux环境下,任何事物都以文件的形式存在,通过文件不仅仅可以访问常规数据,还可以访 ...

  4. 关于 Cocoa Pods 的使用

    前提:电脑上已经安装好CocoaPods. 创建Podfile: 1.进入到项目的目录中: $ cd 目录 2.使用终端命令行: $ vim Podfile 3.然后按i键,进入编辑模式 我现在使用的 ...

  5. MVVM之View和ViewModel的关联

    概要: 将所有的VM在加载到Application的Static Resource中,然后在View中用标签指定. 实现: 1)采用特性指定要添加到StaticResource中的对象 public  ...

  6. day01-day04总结- Python 数据类型及其用法

    Python 数据类型及其用法: 本文总结一下Python中用到的各种数据类型,以及如何使用可以使得我们的代码变得简洁. 基本结构 我们首先要看的是几乎任何语言都具有的数据类型,包括字符串.整型.浮点 ...

  7. .NET中TextBox控件设置ReadOnly=true后台取不到值三种解决方法

    当TextBox设置了ReadOnly=true后要是在前台为控件添加了值,后台是取不到的,值为空,多么郁闷的一个问题经过尝试,发现可以通过如下的方式解决这个问题.感兴趣的朋友可以了解下 当TextB ...

  8. CakePHP之请求与响应对象

    请求与响应对象 请求与响应对象在 CakePHP 2.0 是新增加的.在之前的版本中,这两个对象是由数组表示的,而相关的方法是分散在RequestHandlerComponent,Router,Dis ...

  9. 一款js、css压缩工具yuicompressor

    //压缩JS java -jar yuicompressor-.jar --type js --charset utf- -v src.js > packed.js //压缩CSS java - ...

  10. windows下apache+php+mysql配置

    Apache 2.4.10(文件:httpd-2.4.10-win64-VC11.zip) php 5.6.26 (文件:php-5.6.25-Win32-VC11-x64.zip) mysql 5. ...