hdu 5442 (ACM-ICPC2015长春网络赛F题)
题意:给出一个字符串,长度是2*10^4。将它首尾相接形成环,并在环上找一个起始点顺时针或逆时针走一圈,求字典序最大的走法,如果有多个答案则找起始点最小的,若起始点也相同则选择顺时针。
分析:后缀数组。
首先,需要补充后缀数组的基本知识的话,看这个链接。
http://www.cnblogs.com/rainydays/archive/2011/05/09/2040993.html
我利用这道题重新整理了后缀数组的模板如下:
const int MAX_LEN = ;
//call init_RMQ(f[], n) first.
//then call query(a, b) to quest the RMQ of [a, b].
int power[];
int st[MAX_LEN * ][];
int ln[MAX_LEN * ]; //returns the index of the first minimum value in [x, y]
void init_RMQ(int f[], int n)
{
int i, j;
for (power[] = , i = ; i < ; i++)
{
power[i] = * power[i - ];
}
for (i = ; i < n; i++)
{
st[i][] = i;
}
ln[] = -;
for (int i = ; i <= n; i++)
{
ln[i] = ln[i >> ] + ;
}
for (j = ; j < ln[n]; j++)
{
for (i = ; i < n; i++)
{
if (i + power[j - ] - >= n)
{
break;
}
//for maximum, change ">" to "<"
//for the last, change "<" or ">" to "<=" or ">="
if (f[st[i][j - ]] > f[st[i + power[j - ]][j - ]])
{
st[i][j] = st[i + power[j - ]][j - ];
}
else
{
st[i][j] = st[i][j - ];
}
}
}
} int query(int f[], int x, int y)
{
if(x > y)
{
swap(x, y);
}
int k = ln[y - x + ];
//for maximum, change ">" to "<"
//for the last, change "<" or ">" to "<=" or ">="
if (f[st[x][k]] > f[st[y - power[k] + ][k]])
return st[y - power[k] + ][k];
return st[x][k];
} //first use the constructed function
//call function lcp(l, r) to get the longest common prefix of sa[l] and sa[r]
//have access to the member of sa, myrank, height and so on
//height is the LCP of sa[i] and sa[i + 1]
class SuffixArray
{
public:
char* s;
int n, sa[MAX_LEN], height[MAX_LEN], myrank[MAX_LEN];
int tmp[MAX_LEN], top[MAX_LEN]; SuffixArray()
{} //the string and the length of the string
SuffixArray(char* st, int len)
{
s = st;
n = len + ;
make_sa();
make_lcp();
} void make_sa()
{
// O(N * log N)
int na = (n < ? : n);
memset(top, , na * sizeof(int));
for (int i = ; i < n ; i++)
top[myrank[i] = s[i] & 0xff]++;
for (int i = ; i < na; i++)
top[i] += top[i - ];
for (int i = ; i < n ; i++)
sa[--top[ myrank[i]]] = i;
int x;
for (int len = ; len < n; len <<= )
{
for (int i = ; i < n; i++)
{
x = sa[i] - len;
if (x < )
x += n;
tmp[top[myrank[x]]++] = x;
}
sa[tmp[top[] = ]] = x = ;
for (int i = ; i < n; i++)
{
if (myrank[tmp[i]] != myrank[tmp[i-]] ||
myrank[tmp[i]+len]!=myrank[tmp[i-]+len])
top[++x] = i;
sa[tmp[i]] = x;
}
memcpy(myrank, sa , n * sizeof(int));
memcpy(sa , tmp, n * sizeof(int));
if (x >= n - )
break;
}
} void make_lcp()
{
// O(4 * N)
int i, j, k;
for (j = myrank[height[i = k = ] = ]; i < n - ; i++, k++)
{
while (k >= && s[i] != s[sa[j - ] + k])
{
height[j - ] = (k--);
j = myrank[sa[j] + ];
}
}
init_RMQ(height, n - );
} int lcp(int l, int r)
{
return height[query(height, l, r - )];
}
};
先将两个原字符串拼成一个长度是2倍的字符串,便于处理越过原串末尾的情况。
运行后缀数组求顺时针解。
反转这个长字符串,再运行后缀数组,求逆时针解。
比较两个解,得出最终解。
下面我们来分析一下,每次运行后缀数组之后是如何求解的。
设原串长度为len。
首先从sa[len*2]开始依次向前找。即从排序最大的开始向前找。之所以排序最大的不一定是我们要的答案是因为:
1.它可能起始点在后面的附加的串上,并没有构成完整的一圈。
2.我们还要找起始点最小的答案呢。(当然,对于反转后的字符串,我们要找起始点最大的)
在第一次遇到一个起始点小于len的时候,表明这个可能就是答案。但是前面可能有起始点更小的,我们还要继续沿着sa向前找,但找的时候一定要保证字典序是最大的,所以每次向前移动一个,都要观察height(i, i+1)是否>len,如果不是,则说明字典序已经不是最大了。
特别注意,height==len的情况也是不可以的,因为两者相等很有可能表明其中一个的起始点已经等于len。当然也可以对每个都判断一下起始点位置。
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std; #define d(x) const int MAX_N = (int)(4e4) + ; //call init_RMQ(f[], n) first.
//then call query(a, b) to quest the RMQ of [a, b].
int power[];
int st[MAX_N * ][];
int ln[MAX_N * ]; //returns the index of the first minimum value in [x, y]
void init_RMQ(int f[], int n)
{
int i, j;
for (power[] = , i = ; i < ; i++)
{
power[i] = * power[i - ];
}
for (i = ; i < n; i++)
{
st[i][] = i;
}
ln[] = -;
for (int i = ; i <= n; i++)
{
ln[i] = ln[i >> ] + ;
}
for (j = ; j < ln[n]; j++)
{
for (i = ; i < n; i++)
{
if (i + power[j - ] - >= n)
{
break;
}
//for maximum, change ">" to "<"
//for the last, change "<" or ">" to "<=" or ">="
if (f[st[i][j - ]] > f[st[i + power[j - ]][j - ]])
{
st[i][j] = st[i + power[j - ]][j - ];
}
else
{
st[i][j] = st[i][j - ];
}
}
}
} int query(int f[], int x, int y)
{
if(x > y)
{
swap(x, y);
}
int k = ln[y - x + ];
//for maximum, change ">" to "<"
//for the last, change "<" or ">" to "<=" or ">="
if (f[st[x][k]] > f[st[y - power[k] + ][k]])
return st[y - power[k] + ][k];
return st[x][k];
} //first use the constructed function
//call function lcp(l, r) to get the longest common prefix of sa[l] and sa[r]
//have access to the member of sa, myrank, height and so on
//height is the LCP of sa[i] and sa[i + 1]
class SuffixArray
{
public:
char* s;
int n, sa[MAX_N], height[MAX_N], myrank[MAX_N];
int tmp[MAX_N], top[MAX_N]; SuffixArray()
{} //the string and the length of the string
SuffixArray(char* st, int len)
{
s = st;
n = len + ;
make_sa();
make_lcp();
} void make_sa()
{
// O(N * log N)
int na = (n < ? : n);
memset(top, , na * sizeof(int));
for (int i = ; i < n ; i++)
top[myrank[i] = s[i] & 0xff]++;
for (int i = ; i < na; i++)
top[i] += top[i - ];
for (int i = ; i < n ; i++)
sa[--top[ myrank[i]]] = i;
int x;
for (int len = ; len < n; len <<= )
{
for (int i = ; i < n; i++)
{
x = sa[i] - len;
if (x < )
x += n;
tmp[top[myrank[x]]++] = x;
}
sa[tmp[top[] = ]] = x = ;
for (int i = ; i < n; i++)
{
if (myrank[tmp[i]] != myrank[tmp[i-]] ||
myrank[tmp[i]+len]!=myrank[tmp[i-]+len])
top[++x] = i;
sa[tmp[i]] = x;
}
memcpy(myrank, sa , n * sizeof(int));
memcpy(sa , tmp, n * sizeof(int));
if (x >= n - )
break;
}
} void make_lcp()
{
// O(4 * N)
int i, j, k;
for (j = myrank[height[i = k = ] = ]; i < n - ; i++, k++)
{
while (k >= && s[i] != s[sa[j - ] + k])
{
height[j - ] = (k--);
j = myrank[sa[j] + ];
}
}
init_RMQ(height, n - );
} int lcp(int l, int r)
{
return height[query(height, l, r - )];
}
}; SuffixArray suffix;
int len;
char ans1[MAX_N], ans2[MAX_N];
int pos1, pos2;
char s[MAX_N]; int work1(char* ans)
{
int p = len * ;
while (suffix.sa[p] >= len)
p--;
for (int i = p - ; i >= ; i--)
{
if (suffix.lcp(i, i + ) <= len)
break;
if (suffix.sa[i] < suffix.sa[p])
{
p = i;
}
}
memcpy(ans, s + suffix.sa[p], len);
ans[len] = ;
return suffix.sa[p];
} int work2(char* ans)
{
int p = len * ;
while (suffix.sa[p] >= len)
p--;
for (int i = p - ; i >= ; i--)
{
if (suffix.lcp(i, i + ) <= len)
break;
if (suffix.sa[i] > suffix.sa[p])
{
p = i;
}
}
memcpy(ans, s + suffix.sa[p], len);
ans[len] = ;
return suffix.sa[p];
} int main()
{
int t;
scanf("%d", &t);
while (t--)
{
scanf("%d", &len);
scanf("%s", s);
for (int i = ; i < len; i++)
{
s[len + i] = s[i];
}
suffix = SuffixArray(s, len * );
pos1 = work1(ans1);
reverse(s, s + len * );
suffix = SuffixArray(s, len * );
pos2 = work2(ans2);
pos2 = len - - pos2; if (strcmp(ans1, ans2) > )
{
printf("%d 0\n", pos1 + );
continue;
}else if (strcmp(ans1, ans2) < )
{
printf("%d 1\n", pos2 + );
continue;
}else if (pos1 <= pos2)
{
printf("%d 0\n", pos1 + );
continue;
}else
{
printf("%d 1\n", pos2 + );
}
}
return ;
}
hdu 5442 (ACM-ICPC2015长春网络赛F题)的更多相关文章
- 2013 ACM/ICPC 长春网络赛F题
题意:两个人轮流说数字,第一个人可以说区间[1~k]中的一个,之后每次每人都可以说一个比前一个人所说数字大一点的数字,相邻两次数字只差在区间[1~k].谁先>=N,谁输.问最后是第一个人赢还是第 ...
- hdu 5441 (2015长春网络赛E题 带权并查集 )
n个结点,m条边,权值是 从u到v所花的时间 ,每次询问会给一个时间,权值比 询问值小的边就可以走 从u到v 和从v到u算不同的两次 输出有多少种不同的走法(大概是这个意思吧)先把边的权值 从小到大排 ...
- 2013 ACM/ICPC 长春网络赛E题
题意:给出一个字符串,要从头.尾和中间找出三个完全相等的子串,这些串覆盖的区间互相不能有重叠部分.头.尾的串即为整个字符串的前缀和后缀.问这个相同的子串的最大长度是多少. 分析:利用KMP算法中的ne ...
- 2013 ACM/ICPC 南京网络赛F题
题意:给出一个4×4的点阵,连接相邻点可以构成一个九宫格,每个小格边长为1.从没有边的点阵开始,两人轮流向点阵中加边,如果加入的边构成了新的边长为1的小正方形,则加边的人得分.构成几个得几分,最终完成 ...
- ACM-ICPC 2019南昌网络赛F题 Megumi With String
ACM-ICPC 南昌网络赛F题 Megumi With String 题目描述 给一个长度为\(l\)的字符串\(S\),和关于\(x\)的\(k\)次多项式\(G[x]\).当一个字符串\(str ...
- HDU 4764 Stone (2013长春网络赛,水博弈)
Stone Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submi ...
- HDU 4762 Cut the Cake (2013长春网络赛1004题,公式题)
Cut the Cake Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Tota ...
- HDU 4759 Poker Shuffle(2013长春网络赛1001题)
Poker Shuffle Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Tot ...
- HDU 4768 Flyer (2013长春网络赛1010题,二分)
Flyer Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submi ...
随机推荐
- phpcms使用细节
1.在模板中使用php语句 <?php for ($i=0; $i < 10; $i++) { echo $i."#######<br>"; }?& ...
- 将图片部署在tomcat/iportWork/uploadFiles中
将图片部署在tomcat/iportWork/uploadFiles中 1.在将运行的tomcat目录下创建个二级目录iportWork\uploadFiles,如下图:
- mybatis map foreach遍历
mybatis map foreach遍历 转至http://www.cnblogs.com/yg_zhang/p/4314602.html mybatis 遍历map实例 map 数据如下 Map& ...
- 获取Executor提交的并发执行的任务返回结果的两种方式/ExecutorCompletionService使用
当我们通过Executor提交一组并发执行的任务,并且希望在每一个任务完成后能立即得到结果,有两种方式可以采取: 方式一: 通过一个list来保存一组future,然后在循环中轮训这组future,直 ...
- 利用ant脚本 自动构建svn增量/全量 系统程序升级包
首先请允许我这样说,作为开发或测试,你一定要具备这种 本领.你可以手动打包.部署你的工程,但这不是最好的方法.最好的方式就是全自动化的方式.开发人员提交了代码后,可以自动构建.打包.部署到测试环境. ...
- [Js/Jquery]jquery插件开发
摘要 上篇文章简单学习了js自调用方法.今天就趁热打铁,学一学怎么编写一个jquery插件. JQuery 参考地址:http://www.cnblogs.com/playerlife/archive ...
- 用HTML/JS/PHP方式实现页面延时跳转
WEB开发中经常会遇到页面跳转或延时跳转的需求,掌握各种页面跳转方式非常必要. 以下是我总结有用HTML/JS/PHP三类方式实现跳转的方法,例子皆为三秒后跳转到index.php 页面. 1,HTM ...
- MySql循环插入数据(定义了存储过程)
MySQL一窍不通啊,今天工作上需要用到,请教了别人,做以备忘 DROP PROCEDURE test_insert ; DELIMITER ;; CREATE PROCEDURE test_inse ...
- 面试集锦-常量,const, const 对指针的影响
在C语言中不可改变的数据(量)就是常量 在C语言中有三种常量 字面量(直接量),就是直接写出来的,从写法上就可以看出值与类型等,例如:19,123.456等 名字常量 ...
- ProgressBar样式(转)
普通圆形ProgressBar 该类型进度条也就是一个表示运转的过程,例如发送短信,连接网络等等,表示一个过程正在执行中. 一般只要在XML布局中定义就可以了. <progressBar and ...