[BZOJ 1692] [Usaco2007 Dec] 队列变换 【后缀数组 + 贪心】
---恢复内容开始---
题目链接:BZOJ - 1692
题目分析
首先,有个比较简单的贪心思路:如果当前剩余字符串的两端字母不同,就选取小的字母,这样显然是正确的。
然而若两端字母相同,我们怎么选取呢?
这时我们要从两端分别向内部比较,看那一端向内的字符串字典序小。
比如这个字符串 ABCDBA,从左端向内是 ABC.. 从右端向内是 ABD... 所以就选取左端的字符。
这样直接比较是 O(n^2) 的,我们可以使用后缀数组的 Rank 数组来比较。
我们在字符串后加上分隔符,然后再将字符串反转接在后面,求后缀数组的 Rank 数组。
这样就可以快速比较一个前缀,一个后缀的字典序大小了,具体见代码。
比如 ABCDBA ,就存成 ABCDBA#ABDCBA 。
代码
- #include <iostream>
- #include <cstdio>
- #include <cstdlib>
- #include <cstring>
- #include <cmath>
- #include <algorithm>
- using namespace std;
- const int MaxL = 60000 + 15;
- int n;
- int A[MaxL], Rank[MaxL], SA[MaxL];
- int VA[MaxL], VB[MaxL], VC[MaxL], Sum[MaxL];
- char S[MaxL], Sout[MaxL];
- inline bool Cmp(int *a, int x, int y, int l) {
- return (a[x] == a[y]) && (a[x + l] == a[y + l]);
- }
- void DA(int *A, int n, int m) {
- int *x, *y, *t;
- x = VA; y = VB;
- for (int i = 1; i <= m; ++i) Sum[i] = 0;
- for (int i = 1; i <= n; ++i) ++Sum[x[i] = A[i]];
- for (int i = 2; i <= m; ++i) Sum[i] += Sum[i - 1];
- for (int i = n; i >= 1; --i) SA[Sum[x[i]]--] = i;
- int p, q;
- p = 0;
- for (int j = 1; p < n; j <<= 1, m = p) {
- q = 0;
- for (int i = n - j + 1; i <= n; ++i) y[++q] = i;
- for (int i = 1; i <= n; ++i) {
- if (SA[i] <= j) continue;
- y[++q] = SA[i] - j;
- }
- for (int i = 1; i <= n; ++i) VC[i] = x[y[i]];
- for (int i = 1; i <= m; ++i) Sum[i] = 0;
- for (int i = 1; i <= n; ++i) ++Sum[VC[i]];
- for (int i = 2; i <= m; ++i) Sum[i] += Sum[i - 1];
- for (int i = n; i >= 1; --i) SA[Sum[VC[i]]--] = y[i];
- t = x; x = y; y = t;
- x[SA[1]] = 1; p = 1;
- for (int i = 2; i <= n; ++i)
- x[SA[i]] = Cmp(y, SA[i], SA[i - 1], j) ? p : ++p;
- }
- for (int i = 1; i <= n; ++i) Rank[SA[i]] = i;
- }
- int main()
- {
- scanf("%d", &n);
- for (int i = 1; i <= n; ++i) {
- cin >> S[i];
- A[i] = S[i] - 'A' + 1;
- A[2 * n - i + 2] = A[i];
- }
- A[n + 1] = 27;
- A[n * 2 + 2] = 28;
- DA(A, n * 2 + 2, 28);
- int l = 1, r = n, Top = 0;
- while (l <= r) {
- if (S[l] != S[r]) {
- if (S[l] < S[r]) {
- Sout[++Top] = S[l];
- ++l;
- }
- else {
- Sout[++Top] = S[r];
- --r;
- }
- continue;
- }
- if (l == r) {
- Sout[++Top] = S[l];
- break;
- }
- if (Rank[l] < Rank[n * 2 - r + 2]) {
- Sout[++Top] = S[l];
- ++l;
- }
- else {
- Sout[++Top] = S[r];
- --r;
- }
- }
- for (int i = 1; i <= Top; ++i) {
- printf("%c", Sout[i]);
- if (i % 80 == 0) printf("\n");
- }
- return 0;
- }
---恢复内容结束---
[BZOJ 1692] [Usaco2007 Dec] 队列变换 【后缀数组 + 贪心】的更多相关文章
- BZOJ 1692: [Usaco2007 Dec]队列变换 [后缀数组 贪心]
1692: [Usaco2007 Dec]队列变换 Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 1383 Solved: 582[Submit][St ...
- BZOJ 1692: [Usaco2007 Dec]队列变换 (后缀数组/二分+Hash)
跟BZOJ 4278: [ONTAK2015]Tasowanie一模一样 SA的做法就是把原串倒过来接在原串后面,O(nlogn)O(nlogn)O(nlogn)做后缀数组,就能O(1)O(1)O(1 ...
- 1692: [Usaco2007 Dec]队列变换|后缀数组|贪心
将字符串翻转后接到原串的后面,中间加一个分隔符,每次都贪心选择rankrank小的那个 事实上就是练习一发后缀数组的模板 #include<algorithm> #include<i ...
- 【BZOJ1692】[Usaco2007 Dec]队列变换 后缀数组+贪心
[BZOJ1692][Usaco2007 Dec]队列变换 Description FJ打算带他的N(1 <= N <= 30,000)头奶牛去参加一年一度的“全美农场主大奖赛”.在这场比 ...
- BZOJ 1692: [Usaco2007 Dec]队列变换( 贪心 )
数据 n <= 30000 , 然后 O( n² ) 的贪心也过了..... USACO 数据是有多弱啊 = = ( ps : BZOJ 1640 和此题一模一样 , 双倍经验 ) ------ ...
- ●BZOJ 1692 [Usaco2007 Dec]队列变换
题链: http://www.lydsy.com/JudgeOnline/problem.php?id=1692 题解: 后缀数组,贪心由于每次可以取出旧队列的首部或尾部放在新队列的尾部.所以就需要比 ...
- BZOJ 1692: [Usaco2007 Dec]队列变换
Description FJ打算带他的N(1 <= N <= 30,000)头奶牛去参加一年一度的"全美农场主大奖赛".在这场比赛中,每个参赛者都必须让他的奶牛排成一列 ...
- bzoj:1692 [Usaco2007 Dec]队列变换&&1640 [Usaco2007 Nov]Best Cow Line 队列变换
Description FJ打算带他的N(1 <= N <= 30,000)头奶牛去参加一年一度的“全美农场主大奖赛”.在这场比赛中,每个参赛者都必须让他的奶牛排成一列,然后领她们从裁判席 ...
- bzoj 1692: [Usaco2007 Dec]队列变换 ——二分+hash
Description FJ打算带他的N(1 <= N <= 30,000)头奶牛去参加一年一度的“全美农场主大奖赛”.在这场比赛中,每个参赛者都必须让他的奶牛排成一列,然后领她们从裁判席 ...
随机推荐
- richTextBox1 转到行号
private void button2_Click(object sender, EventArgs e) { Win32CommonDialog.frm_GOTO frm = new ...
- [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 ...
- linux lsof命令详解
linux lsof命令详解 简介 lsof(list open files)是一个列出当前系统打开文件的工具.在linux环境下,任何事物都以文件的形式存在,通过文件不仅仅可以访问常规数据,还可以访 ...
- 关于 Cocoa Pods 的使用
前提:电脑上已经安装好CocoaPods. 创建Podfile: 1.进入到项目的目录中: $ cd 目录 2.使用终端命令行: $ vim Podfile 3.然后按i键,进入编辑模式 我现在使用的 ...
- MVVM之View和ViewModel的关联
概要: 将所有的VM在加载到Application的Static Resource中,然后在View中用标签指定. 实现: 1)采用特性指定要添加到StaticResource中的对象 public ...
- day01-day04总结- Python 数据类型及其用法
Python 数据类型及其用法: 本文总结一下Python中用到的各种数据类型,以及如何使用可以使得我们的代码变得简洁. 基本结构 我们首先要看的是几乎任何语言都具有的数据类型,包括字符串.整型.浮点 ...
- .NET中TextBox控件设置ReadOnly=true后台取不到值三种解决方法
当TextBox设置了ReadOnly=true后要是在前台为控件添加了值,后台是取不到的,值为空,多么郁闷的一个问题经过尝试,发现可以通过如下的方式解决这个问题.感兴趣的朋友可以了解下 当TextB ...
- CakePHP之请求与响应对象
请求与响应对象 请求与响应对象在 CakePHP 2.0 是新增加的.在之前的版本中,这两个对象是由数组表示的,而相关的方法是分散在RequestHandlerComponent,Router,Dis ...
- 一款js、css压缩工具yuicompressor
//压缩JS java -jar yuicompressor-.jar --type js --charset utf- -v src.js > packed.js //压缩CSS java - ...
- 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. ...