思路

后缀数组

先都拼在一起

二分+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的更多相关文章

  1. UVA11107 Life Forms --- 后缀数组

    UVA11107 Life Forms 题目描述: 求出出现在一半以上的字符串内的最长字符串. 数据范围: \(\sum len(string) <= 10^{5}\) 非常坑的题目. 思路非常 ...

  2. UVA11107 Life Forms SA模板

    Life Forms Time Limit: 5000MS   Memory Limit: 65536K Total Submissions: 16827   Accepted: 4943 Descr ...

  3. UVA-11107 Life Forms(后缀数组)

    题目大意:给出n个字符串,找出所有最长的在超过一半的字符串中出现的子串. 题目分析:将所有的字符串连成一个,二分枚举长度,每次用O(n)的时间复杂度判断.连接字符串的时候中间添一个没有出现过的字符. ...

  4. UVA-11107 Life Forms(求出现K次的子串,后缀数组+二分答案)

    题解: 题意: 输入n个DNA序列,你的任务是求出一个长度最大的字符串,使得它在超过一半的DNA序列中出现.如果有多解,按照字典序从小到大输入所有解. 把n个DNA序列拼在一起,中间用没有出现过的字符 ...

  5. 【UVA11107 训练指南】Life Forms【后缀数组】

    题意 输入n(n<=100)个字符串,每个字符串长度<=1000,你的任务是找出一个最长的字符串使得超过一半的字符串都包含这个字符串. 分析 训练指南上后缀数组的一道例题,据说很经典(估计 ...

  6. Wizard Framework:一个自己开发的基于Windows Forms的向导开发框架

    最近因项目需要,我自己设计开发了一个基于Windows Forms的向导开发框架,目前我已经将其开源,并发布了一个NuGet安装包.比较囧的一件事是,当我发布了NuGet安装包以后,发现原来已经有一个 ...

  7. xamarin.forms新建项目android编译错误

    vs2015 update3 新建的xamarin.forms项目中的android项目编译错误.提示缺少android_m2repository_r22.zip,96659D653BDE0FAEDB ...

  8. ASP.NET Forms 身份验证

    ASP.NET Forms 身份验证 在开发过程中,我们需要做的事情包括: 1. 在 web.config 中设置 Forms 身份验证相关参数.2. 创建登录页. 登录页中的操作包括: 1. 验证用 ...

  9. Xamarin.Forms 简介

    An Introduction to Xamarin.Forms 来源:http://developer.xamarin.com/guides/cross-platform/xamarin-forms ...

随机推荐

  1. LeetCode 171 Excel Sheet Column Number 解题报告

    题目要求 Given a column title as appear in an Excel sheet, return its corresponding column number. For e ...

  2. WinAPI 字符及字符串函数(15): CharNext、CharPrev

    unit Unit1; interface uses   Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, For ...

  3. Mybatis+Mysql逆向工程

    目录结构: pom文件: <?xml version="1.0" encoding="UTF-8"?> <project xmlns=&quo ...

  4. macOS 版微信小助手,支持微信多开、防撤回、远程控制mac、自动回复等等

    微信小助手 GitHub大牛提供的微信小助手是一款插件,该插件具备多开.防撤回.免手机认证登录.自动回复.远程控制自己的 macOS.群发等众多功能 GitHub网址:https://github.c ...

  5. Xpath语法详解

    1.简介 XPath是一门在XML和HTML文档中查找信息的语言,可以用来在XML和HTML文档中对元素和属性进行遍历 XPath的安装 Chrome插件XPath Helper 点Chrome浏览器 ...

  6. 【Linux】Set CentOS no GUI default

    [root@localhost ~]#vi /etc/inittab 重点看下面这部分代码 #   0 - halt (Do NOT set initdefault to this)          ...

  7. JAVA获取汉字拼音首字母

    package com.common.util; import java.io.UnsupportedEncodingException; /** * 取得给定汉字串的首字母串,即声母串 * Titl ...

  8. Caused by: com.mysql.jdbc.exceptions.jdbc4.CommunicationsException: Communications link failure

    很长的报错,截取   ERROR c.a.d.p.DruidDataSource - discard connection   com.mysql.jdbc.exceptions.jdbc4.Comm ...

  9. redis----------linux和mac如何安装redis和启动,关闭

    1.打开官网https://redis.io/download.官网有安装命令 2.以下是我的执行过程截图 执行完官网给的命令以后,再执行  make PREFIX=/usr/local/redis ...

  10. cygwin 安装 apt-cyg

    apt-cyg apt-cyg is a Cygwin package manager. It includes a command-line installer for Cygwin which c ...