spoj 1812 lcsII (后缀自动机)
题意:求多个串的lcs,最多10个串,每个串最长10w
解题思路:后缀自动机。先建好第一个串的sam,然后后面的串拿上去跑(这个过程同前一题)。sam上,节点要记录两个信息,先说mi[p],其意思是p节点能匹配的所有串的最短的长度是多少,那我们如何更新这个mi呢?于是我们要记录另一个信息,mm[p]表示,跑当前串时,所跑到的节点能匹配的最大值,用这个最大值去更新mi。一个串跑完之后,要根据parent树,把mm往fa更新,然后用底下传上来的mm更新mi。怎么更新parent呢?很简单,我们知道,parent tree中父亲的代表串的长度必然比儿子短,所以我们只要根据代表串的长度排序,然后从后往前更新即可。
这题我有个小疑问,我们给mi赋初值时,用INF为何不可?而要用当前节点的val值呢?还望有知道的大神能不吝赐教啊。
#include<stdio.h>
#include<string.h>
#include<algorithm>
using namespace std ; const int maxn = 311111 ; int max ( int a , int b ) { return a > b ? a : b ; }
int min ( int a , int b ) { return a < b ? a : b ; }
int wv[maxn<<1] , ws[maxn<<1] ;
int mm[maxn<<1] , mi[maxn<<1] ;
struct sam {
int fa[maxn<<1] , c[26][maxn<<1] , val[maxn<<1] , pos[maxn<<1] ;
int tot , last ;
inline int new_node ( int step ) {
val[++tot] = step ;
int i ;
for ( i = 0 ; i < 26 ; i ++ ) c[i][tot] = 0 ;
fa[tot] = 0 , mi[tot] = val[tot] ;
return tot ;
}
void add ( int k ) {
int p = last , i ;
int np = new_node ( val[p] + 1 ) ;
while ( p && !c[k][p] ) c[k][p] = np , p = fa[p] ;
if ( !p ) fa[np] = 1 ;
else {
int q = c[k][p] ;
if ( val[p] + 1 == val[q] ) fa[np] = q ;
else {
int nq = new_node ( val[p] + 1 ) ;
for ( i = 0 ; i < 26 ; i ++ ) c[i][nq] = c[i][q] ;
fa[nq] = fa[q] ;
fa[q] = fa[np] = nq ;
while ( p && c[k][p] == q ) c[k][p] = nq , p = fa[p] ;
}
}
last = np ;
}
void build ( char *s ) {
tot = 0 ;
last = new_node ( 0 ) ;
int len = strlen ( s ) , i ;
for ( i = 0 ; i < len ; i ++ ) add ( s[i] - 'a' ) ;
for ( i = 1 ; i <= tot ; i ++ ) wv[val[i]] ++ ;
for ( i = 1 ; i <= tot ; i ++ ) wv[i] += wv[i-1] ;
for ( i = 1 ; i <= tot ; i ++ ) ws[wv[val[i]]--] = i ;
}
void work ( char *s ) {
int len = strlen ( s ) ;
int p = 1 , i , temp = 0 ;
for ( i = 0 ; i < len ; i ++ ) {
int k = s[i] - 'a' ;
if ( c[k][p] ) temp ++ , p = c[k][p] ;
else {
while ( p && !c[k][p] ) p = fa[p] ;
if ( !p ) temp = 0 , p = 1 ;
else temp = val[p] + 1 , p = c[k][p] ;
}
mm[p] = max ( mm[p] , temp ) ;
}
for ( i = tot ; i >= 1 ; i -- ) {
p = ws[i] ;
mi[p] = min ( mi[p] , mm[p] ) ;
if ( fa[p] && mm[fa[p]] < mm[p] ) mm[fa[p]] = mm[p] ;
mm[p] = 0 ;
}
}
int solve () {
int i , ans = 0 ;
for ( i = 1 ; i <= tot ; i ++ )
ans = max ( ans , mi[i]) ;
return ans ;
}
} suf ;
char s[maxn] ;
int main () {
// freopen ( "a.txt" , "r" , stdin ) ;
scanf ( "%s" , s ) ;
suf.build ( s ) ;
int cnt = 0 ;
while ( scanf ( "%s" , s ) != EOF ) {
suf.work ( s ) ;
// cnt ++ ;
// if ( cnt == 2 ) break ;
}
printf ( "%d\n" , suf.solve () ) ;
}
/*
skds
fkds
aajfaa
*/
spoj 1812 lcsII (后缀自动机)的更多相关文章
- SPOJ.1812.LCS2(后缀自动机)
		
题目链接 \(Description\) 求最多10个串的LCS(最长公共子序列). \(Solution\) 类比上题,对一个串建SAM,我们可以逐串地求出其在每个节点所能匹配的最大长度mx[i]. ...
 - SPOJ 1812 LCS2 [后缀自动机 DP]
		
题意: 求多个串<=10的最长连续子串 一个串建SAM,然后其他串在上面走 每个状态记录所有串在这个状态的公共子串的最小值 一个串在上面走的时候记录与每个状态公共子串的最大值,注意出现次数向父亲 ...
 - POJ.2774.Long Long Message/SPOJ.1811.LCS(后缀自动机)
		
题目链接 POJ2774 SPOJ1811 LCS - Longest Common Substring 确实比后缀数组快多了(废话→_→). \(Description\) 求两个字符串最长公共子串 ...
 - SPOJ NSUBSTR Substrings 后缀自动机
		
人生第一道后缀自动机,总是值得纪念的嘛.. 后缀自动机学了很久很久,先是看CJL的论文,看懂了很多概念,关于right集,关于pre,关于自动机的术语,关于为什么它是线性的结点,线性的连边.许多铺垫的 ...
 - 2018.12.15 spoj Substrings(后缀自动机)
		
传送门 后缀自动机基础题. 求长度为iii的子串出现次数的最大值. 对原串建出samsamsam,然后用sizsizsiz更新每个maxlenmaxlenmaxlen的答案. 然后由于后缀链接将其转化 ...
 - SPOJ NSUBSTR Substrings ——后缀自动机
		
建后缀自动机 然后统计次数,只需要算出right集合的大小即可, 然后更新f[l[i]]和rit[i]取个max 然后根据rit集合短的一定包含长的的性质,从后往前更新一遍即可 #include &l ...
 - Lexicographical Substring Search SPOJ - SUBLEX (后缀自动机)
		
Lexicographical Substrings Search \[ Time Limit: 149 ms \quad Memory Limit: 1572864 kB \] 题意 给出一个字符串 ...
 - Substrings SPOJ - NSUBSTR (后缀自动机)
		
Substrings \[ Time Limit: 100ms\quad Memory Limit: 1572864 kB \] 题意 给出一个长度为 \(250000\) 的字符串,求出所有 \(x ...
 - Longest Common Substring II SPOJ - LCS2 (后缀自动机)
		
Longest Common Substring II \[ Time Limit: 236ms\quad Memory Limit: 1572864 kB \] 题意 给出\(n\)个子串,要求这\ ...
 
随机推荐
- BZOJ 3675: [Apio2014]序列分割( dp + 斜率优化 )
			
WA了一版... 切点确定的话, 顺序是不会影响结果的..所以可以dp dp(i, k) = max(dp(j, k-1) + (sumn - sumi) * (sumi - sumj)) 然后斜率优 ...
 - 关于Ubuntu12.04下code::blocks不能使用debug解决方法
			
问题描述: 系统:ubuntu 12.04 code::blocks版本:10.05 问题现象:debug->start 之后出现:warning: GDB: Fail ...
 - Linux内存分析
			
Linux命令----分析内存的瓶颈 为了提高磁盘存取效率, Linux做了一些精心的设计, 除了对dentry进行缓存(用于VFS,加速文件路径名到inode的转换), 还采取了两种主要Cac ...
 - 编程的毛病——C++之父访谈
			
原文见:http://www.technologyreview.com/InfoTech/17831/ 翻译:xeon 11/29/2006 在20世纪的80年代和90年代,Bjarne Strou ...
 - android 仿小米icon处理,加阴影和边框
			
本人自己在做一个launcher,所以须要处理icon,加阴影和边框等.这仅仅是一种处理方法,其它的处理方法类似. 源码: https://github.com/com314159/LauncherI ...
 - Spring IOC和DI原理讲解并制作LazyCoder版的Spring (一)
			
转载请注意出处:http://blog.csdn.net/zcm101 写在前面的话 最近,给项目组成员培训了Spring 控制反转和依赖注入的原理,并自己做了个Lazy Coder版的Spring, ...
 - c++,static  静态成员变量 / 静态成员函数
			
静态成员变量: //静态成员变量(static) // //1.如果想在同类的多个对象之间实现数据共享 ,可以用静态 //成员变量,即用static修饰的成员变量,例 static int a; // ...
 - Android,机器狗应用
			
源码如下: package com.wyl.jqr; import java.io.BufferedReader; import java.io.IOException; import java.io ...
 - Hibernate入门之配置文件
			
<?xml version='1.0' encoding='utf-8'?> <!DOCTYPE hibernate-configuration PUBLIC "-//Hi ...
 - MySQL 出现 The table is full 的解决方法
			
原文链接: MySQL 出现 The table is full 的解决方法 浅谈MySql的存储引擎(表类型) MySQL 出现 The table is full 只有一个原因,对应的表数据容量达 ...