Description

Little cat在Byterland的首都读物理专业。这些天他收到了一条悲伤地信息:他的母亲生病了。担心买火车票花钱太多(Byterland是一个巨大的国家,因此他坐火车回家需要16小时),他决定只给母亲发短信。 
Little cat的家境并不富裕,因此他经常去营业厅查看自己发短信花了多少钱。昨天营业厅的电脑坏掉了,打印出两条很长的信息。机智的little cat很快发现: 
1.信息中所有的字符都是小写英文字母,没有标点和空格。 
2.所有的短信都被连在了一起——第i+1条短信直接接在第i条短信后面——这就是这两条信息如此长的原因。 
3.虽然他发的短信都被连在了一起,但由于电脑坏掉了,它们的左边或右边都可能会有许多冗余字符。 
例如:如果短信是"motheriloveyou",电脑打印出的每条信息都可能是 "hahamotheriloveyou", "motheriloveyoureally", "motheriloveyouornot", "bbbmotheriloveyouaaa",等等。 
4.因为这些乱七八糟的问题,little cat打印了两遍(所以有两条非常长的信息)。尽管原始的短信文本在两条信息中都一样,但两条信息在文本两侧的冗余字符都可能不一样。 
给出这两条很长的信息,输出little cat写下的原始短信文本的最长可能长度。 
背景: 
在Byterland,短信按照美元/字节的单位计价。这就是little cat想要知道原始文本最长可能长度的原因。 
为什么让你写一个程序?有四个原因: 
1.little cat这些天忙于他的物理课程。 
2.little cat不想透露他对母亲说了什么。 
3.POJ是个好网站。 
4.little cat想要从POJ那里挣点钱,并尝试说服他的母亲去医院

Input

两行两个由小写英文字母组成的字符串。字符串长度都不会超过100000

Output

一行一个整数,即little cat写下的原始文本的最长可能长度。

Sample Input

yeshowmuchiloveyoumydearmotherreallyicannotbelieveit 
yeaphowmuchiloveyoumydearmother

Sample Output

27

题解:

将两串合并,中间插入一个'#'保证high不会交叉

直接求出hight数组然后找出满足sa分别在两串的最大hight值

 #include <algorithm>
#include <iostream>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#include <cmath>
using namespace std;
const int N=;
char S1[N],S2[N];int s[N],n,c[N],sa[N],y[N],x[N],rk[N];
bool comp(int i,int j,int k){
return y[i]==y[j] && y[i+k]==y[j+k];
}
void getSA(){
int m=,t=;
for(int i=;i<=m;i++)c[i]=;
for(int i=;i<=n;i++)c[x[i]=s[i]]++;
for(int i=;i<=m;i++)c[i]+=c[i-];
for(int i=n;i>=;i--)sa[c[x[i]]--]=i;
for(int k=;k<=n;k<<=){
t=;
for(int i=;i<=m;i++)y[i]=;
for(int i=n-k+;i<=n;i++)y[++t]=i;
for(int i=;i<=n;i++)if(sa[i]>k)y[++t]=sa[i]-k;
for(int i=;i<=m;i++)c[i]=;
for(int i=;i<=n;i++)c[x[i]]++;
for(int i=;i<=m;i++)c[i]+=c[i-];
for(int i=n;i>=;i--)sa[c[x[y[i]]]--]=y[i];
swap(x,y);
t=x[sa[]]=;
for(int i=;i<=n;i++)x[sa[i]]=comp(sa[i-],sa[i],k)?t:++t;
if(t==n)break;
m=t;
}
for(int i=;i<=n;i++)rk[sa[i]]=i;
}
int high[N];
void gethight(){
int h=,j;
for(int i=;i<=n;i++){
j=sa[rk[i]-];
if(h>)h--;
for(;j+h<=n && i+h<=n;h++)
if(s[i+h]!=s[j+h])break;
high[rk[i]-]=h;
}
}
void work()
{
n=;
scanf("%s%s",S1,S2);
for(int i=,sz=strlen(S1);i<sz;i++)s[++n]=S1[i]-'a'+;
s[++n]=;
for(int i=,sz=strlen(S2);i<sz;i++)s[++n]=S2[i]-'a'+;
getSA();
gethight();
int ans=;int tx=strlen(S1);
for(int i=;i<n;i++)if(high[i]>ans && ((sa[i]<=tx)!=(sa[i+]<=tx)))ans=high[i];
printf("%d\n",ans);
} int main()
{
work();
return ;
}

最新文章

  1. Android 如何判断一个应用在运行(转)
  2. DDD 领域驱动设计-三个问题思考实体和值对象(续)
  3. swift 创建桥接头文件
  4. EF框架step by step(1)—Database-First
  5. 【BZOJ】3028: 食物
  6. Xamarin开发Android笔记:背景操作
  7. PHP 二维数组根据相同的值进行合并
  8. AspectJ本质剖析
  9. java中HashSet详解(转)
  10. 搭建sql注入实验环境(基于windows)
  11. (译)如何在ASP.NET中安全使用ViewState
  12. 一个功能齐全的IOS音乐播放器应用源码
  13. java servlet+jquery+json学习小例子
  14. Net Core- 配置组件
  15. Mac os 进行Android开发笔记(2)
  16. $.data()、$().data
  17. [Leetcode] Binary search, DP--300. Longest Increasing Subsequence
  18. jmeter系列-------注意事项
  19. 千万不要随意在网上下载ojdbcjar包来使用,ORA-01461错误解决
  20. JSON.parse(JSON.stringify(obj))

热门文章

  1. 2017-2018-1 我爱学Java 第六七周 作业
  2. webview缓存及跳转时截取url地址、监听页面变化
  3. 深入分析Java Web中的编码问题
  4. windows系统下安装 node.js (node.js安装及环境配置)
  5. Node入门教程(2)第一章:NodeJS 概述
  6. Mybatis学习日志
  7. Linux--初次体验
  8. ios开发常识(1)开发语言和参考资料
  9. linux下Tab及shell 补全python
  10. 给定了经纬度的一张my_latlng表,和一个my_grid表,怎么实现my_latlng表回mygrid中的id?