UVA11107 Life Forms
思路
后缀数组
先都拼在一起
二分+height分段
按照小于x的为分界,判断是否有一个分段中包含超过n/2个串
代码
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
struct Node{
int pos,r[2];
}x[200100],midx[200100];
int n,sa[200100],ranks[200100],barrel[200100],height[200100],belong[200100],has[210],numx;
char s[200100],c[200100];
int c_sort(int n,int lim){
for(int i=0;i<2;i++){
memset(barrel,0,sizeof(barrel));
for(int j=1;j<=n;j++)
barrel[x[j].r[i]]++;
for(int j=1;j<=lim;j++)
barrel[j]+=barrel[j-1];
for(int j=n;j>=1;j--)
midx[barrel[x[j].r[i]]--]=x[j];
for(int j=1;j<=n;j++)
x[j]=midx[j];
}
ranks[x[1].pos]=1;
int cnt=1;
for(int i=2;i<=n;i++){
if(x[i].r[0]==x[i-1].r[0]&&x[i].r[1]==x[i-1].r[1])
ranks[x[i].pos]=cnt;
else
ranks[x[i].pos]=++cnt;
}
return cnt;
}
void cal_sa(int n){
for(int i=1;i<=n;i++)
x[i]=(Node){i,s[i],0};
int cnt=c_sort(n,255);
for(int i=1;cnt<n;i<<=1){
for(int j=1;j<=n;j++)
x[j]=(Node){j,(i+j<=n)?ranks[i+j]:0,ranks[j]};
cnt=c_sort(n,cnt);
}
for(int i=1;i<=n;i++)
sa[ranks[i]]=i;
for(int i=1,j=0,k;i<=n;height[ranks[i++]]=j)
for(j?j--:0,k=sa[ranks[i]-1];s[i+j]==s[j+k];j++);
}
// void init(void){
// for(int i=1;i<=n;i++)
// ST[i][0]=height[i];
// for(int i=1;i<20;i++)
// for(int j=1;j<=n;j++)
// ST[j][i]=min(ST[j][i-1],ST[min(j+(1<<(i-1)),n+10)][i-1]);
// }
// int query(int l,int r){
// l=ranks[l];
// r=ranks[r];
// l++;
// if(l>r)
// swap(l,r);
// int k=0;
// while((1<<k+1)<=(r-l+1))
// k++;
// return min(ST[l][k],ST[r-(1<<k)+1][k]);
// }
bool check(int x){
// printf("check %d\n",x);
memset(has,0,sizeof(has));
int midnum=0;
has[0]=true;
if(!has[belong[sa[1]]]){
++midnum;
has[belong[sa[1]]]=true;
}
if(midnum>(numx/2)){
return true;
}
for(int i=2;i<=n;i++){
if(height[i]<x){
memset(has,0,sizeof(has));
has[0]=true;
midnum=0;
if(!has[belong[sa[i]]]){
midnum++;
has[belong[sa[i]]]=true;
}
}
else{
if(!has[belong[sa[i]]]){
midnum++;
has[belong[sa[i]]]=true;
}
}
if(midnum>(numx/2)){
return true;
}
}
return false;
}
void print(int x){
memset(has,0,sizeof(has));
int midnum=0,f=1;
has[0]=true;
if(!has[belong[sa[1]]]){
++midnum;
has[belong[sa[1]]]=true;
}
if(midnum>(numx/2)&&f){
f=0;
for(int j=1;j<=x;j++)
putchar(s[sa[1]+j-1]);
putchar('\n');
}
for(int i=2;i<=n;i++){
if(height[i]<x){
f=1;
memset(has,0,sizeof(has));
has[0]=true;
midnum=0;
if(!has[belong[sa[i]]]){
midnum++;
has[belong[sa[i]]]=true;
}
}
else{
if(!has[belong[sa[i]]]){
midnum++;
has[belong[sa[i]]]=true;
}
}
if(midnum>(numx/2)&&f){
f=0;
for(int j=1;j<=x;j++)
putchar(s[sa[i]+j-1]);
putchar('\n');
}
}
}
void init(void){
memset(s,0,sizeof(s));
memset(height,0,sizeof(height));
memset(sa,0,sizeof(sa));
memset(ranks,0,sizeof(ranks));
memset(belong,0,sizeof(belong));
memset(midx,0,sizeof(midx));
memset(x,0,sizeof(x));
}
int main(){
// freopen("test.in","r",stdin);
// freopen("test.out","w",stdout);
int cnt=0;
while(scanf("%d",&numx)==1&&numx){
init();
n=0;
cnt++;
if(cnt>1)
putchar('\n');
if(numx==1){
scanf("%s",s);
printf("%s\n",s);
continue;
}
for(int i=1;i<=numx;i++){
scanf("%s",c+1);
int len=strlen(c+1);
for(int j=1;j<=len;j++){
s[n+j]=c[j];
belong[n+j]=i;
}
n+=len;
s[++n]='z'+i;
belong[n]=0;
}
// for(int i=1;i<=n;i++)
// putchar(s[i]);
// putchar('\n');
cal_sa(n);
// for(int i=1;i<=n;i++){
// printf("%d ",height[i]);
// }
// printf("\n");
// for(int i=1;i<=n;i++){
// printf("%d ",belong[sa[i]]);
// }
// printf("\n");
// printf("ok\n");
int l=0,r=n,ans=0;
while(l<=r){
int mid=(l+r)>>1;
if(check(mid))
l=mid+1,ans=mid;
else
r=mid-1;
}
// printf("ans=%d\n",ans);
if(ans==0){
printf("?\n");
}
else{
print(ans);
}
}
return 0;
}
UVA11107 Life Forms的更多相关文章
- UVA11107 Life Forms --- 后缀数组
UVA11107 Life Forms 题目描述: 求出出现在一半以上的字符串内的最长字符串. 数据范围: \(\sum len(string) <= 10^{5}\) 非常坑的题目. 思路非常 ...
- UVA11107 Life Forms SA模板
Life Forms Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 16827 Accepted: 4943 Descr ...
- UVA-11107 Life Forms(后缀数组)
题目大意:给出n个字符串,找出所有最长的在超过一半的字符串中出现的子串. 题目分析:将所有的字符串连成一个,二分枚举长度,每次用O(n)的时间复杂度判断.连接字符串的时候中间添一个没有出现过的字符. ...
- UVA-11107 Life Forms(求出现K次的子串,后缀数组+二分答案)
题解: 题意: 输入n个DNA序列,你的任务是求出一个长度最大的字符串,使得它在超过一半的DNA序列中出现.如果有多解,按照字典序从小到大输入所有解. 把n个DNA序列拼在一起,中间用没有出现过的字符 ...
- 【UVA11107 训练指南】Life Forms【后缀数组】
题意 输入n(n<=100)个字符串,每个字符串长度<=1000,你的任务是找出一个最长的字符串使得超过一半的字符串都包含这个字符串. 分析 训练指南上后缀数组的一道例题,据说很经典(估计 ...
- Wizard Framework:一个自己开发的基于Windows Forms的向导开发框架
最近因项目需要,我自己设计开发了一个基于Windows Forms的向导开发框架,目前我已经将其开源,并发布了一个NuGet安装包.比较囧的一件事是,当我发布了NuGet安装包以后,发现原来已经有一个 ...
- xamarin.forms新建项目android编译错误
vs2015 update3 新建的xamarin.forms项目中的android项目编译错误.提示缺少android_m2repository_r22.zip,96659D653BDE0FAEDB ...
- ASP.NET Forms 身份验证
ASP.NET Forms 身份验证 在开发过程中,我们需要做的事情包括: 1. 在 web.config 中设置 Forms 身份验证相关参数.2. 创建登录页. 登录页中的操作包括: 1. 验证用 ...
- Xamarin.Forms 简介
An Introduction to Xamarin.Forms 来源:http://developer.xamarin.com/guides/cross-platform/xamarin-forms ...
随机推荐
- linux安装elk
环境: centOS7 JDK8 Elasticsearch-6.6.1 Logstash-6.6.1 Kibana-6.6.1 准备: jdk下载地址:https://www.oracle.com/ ...
- 学号 20175313《Java程序设计》 第七周学习总结
目录 一.教材学习内容总结 二.教材学习中的问题和解决过程 三.代码托管 四.心得体会 五.学习进度条 六.参考资料 一.教材学习内容总结 第八章主要内容 了解String类 常量对象:常量池中的数据 ...
- Lists.newArrayList的一个小坑
把一个用户ID转换成List存储,最开始我使用的方法是: // 用户ID Integer userId = 120; // id 转 List List<integer> userIds ...
- ReLU激活函数的缺点
训练的时候很”脆弱”,很容易就”die”了,训练过程该函数不适应较大梯度输入,因为在参数更新以后,ReLU的神经元不会再有激活的功能,导致梯度永远都是零. 例如,一个非常大的梯度流过一个 ReLU 神 ...
- 2019-oo-第一次总结
一.度量分析程序结构 1.UML类图分析 1.1第一次作业 1.2第二次作业 1.3第三次作业 1.4总结 从UML类图三次作业的可以看出,我从一个类到逐渐利用多个类,代码结构在不 ...
- cocos creator使用anysdk接入admob广告教程
http://lolling787.lofter.com/post/1f5b6553_12925042 cocos creator使用anysdk接入admob广告
- 短信外部浏览器H5链接一键跳转微信打开任意站
今天讲讲微信跳转的那些事情,这项技术最早出现在在线广告上面,可以从外部引流到微信并打开微信内置浏览器然后打开一个指定的网页地址,在这个网页里面可以放任何想推广的内容,可以是引导文案.活动内容,或者是一 ...
- 图表管理账单的NABCD
首先,我们团队的项目目标是记账本.就我个人理解,记账本中心功能有两项,第一,记录:第二,显示.而本篇博客主要描述用各种不同的图表来显示的NABCD. 首先是N(need),用户的需求就是我们的动力!利 ...
- 【转】OJ提交时G++与C++的区别
关于G++ 首先更正一个概念,C++是一门计算机编程语言,G++不是语言,是一款编译器中编译C++程序的命令而已.那么他们之间的区别是什么? 在提交题目中的语言选项里,G++和C++都代表编译的方式. ...
- 解决c# progressBar更新出现界面假死
最近一个项目需求中的一个功能是需要用progressBar反映处理文件的进度. 研究了Invoke和BeginInvoke方法. Control.Invoke 方法 (Delegate) :在拥有此控 ...