poj1509 Glass Beads 找最小的字母边转移
来源:互联网 发布:网络连接图片 编辑:程序博客网 时间:2024/06/05 22:47
传送门
Sample Input
4helloworldamandamandadontcallmebfuaaabaaa
Sample Output
101165
poj1509 Glass Beads
每次找最小的字母边转移,转移l次,找到的就是最小串表示的结尾点。为什么每次找最小的子字母边转移,不会使得在没找到l次后就到达结束状态(没法再转移下去了)呢?
因为,后缀自动机可以串接受所有后缀。如果一个后缀长度小于l,由于开始倍增过,这个后缀一定存在某个长度大于l的后缀使得那个后缀的前缀和这个后缀是一样的,当前状态将能够识别那个后缀,继续转移下去。
好绕口,举例来说,对于l+i这个位置的后缀,i这个位置的后缀的前缀和l+i位置的后缀是一样的,当前状态如果是识别了l+i,那么也一定还能继续识别i。
其实这个过程就是一个筛选后缀的过程。假设有一个集合。起先,集合中包含了所有的后缀,每次找最小的边转移,就是找出后缀中当前字符最小的。
//china no.1#pragma comment(linker, "/STACK:1024000000,1024000000")#include <vector>#include <iostream>#include <string>#include <map>#include <stack>#include <cstring>#include <queue>#include <list>#include <stdio.h>#include <set>#include <algorithm>#include <cstdlib>#include <cmath>#include <iomanip>#include <cctype>#include <sstream>#include <functional>#include <stdlib.h>#include <time.h>#include <bitset>using namespace std;#define pi acos(-1)#define s_1(x) scanf("%d",&x)#define s_2(x,y) scanf("%d%d",&x,&y)#define s_3(x,y,z) scanf("%d%d%d",&x,&y,&z)#define PI acos(-1)#define endl '\n'#define srand() srand(time(0));#define me(x,y) memset(x,y,sizeof(x));#define foreach(it,a) for(__typeof((a).begin()) it=(a).begin();it!=(a).end();it++)#define close() ios::sync_with_stdio(0); cin.tie(0);#define FOR(x,n,i) for(int i=x;i<=n;i++)#define FOr(x,n,i) for(int i=x;i<n;i++)#define fOR(n,x,i) for(int i=n;i>=x;i--)#define fOr(n,x,i) for(int i=n;i>x;i--)#define W while#define sgn(x) ((x) < 0 ? -1 : (x) > 0)#define bug printf("***********\n");#define db double#define ll long long#define mp make_pair#define pb push_backtypedef pair<long long int,long long int> ii;typedef long long LL;const int INF=0x3f3f3f3f;const LL LINF=0x3f3f3f3f3f3f3f3fLL;const int dx[]={-1,0,1,0,1,-1,-1,1};const int dy[]={0,1,0,-1,-1,1,-1,1};const int maxn=1e5+10;const int maxx=600005;const double EPS=1e-8;const double eps=1e-8;const int mod=1e9+7;template<class T>inline T min(T a,T b,T c) { return min(min(a,b),c);}template<class T>inline T max(T a,T b,T c) { return max(max(a,b),c);}template<class T>inline T min(T a,T b,T c,T d) { return min(min(a,b),min(c,d));}template<class T>inline T max(T a,T b,T c,T d) { return max(max(a,b),max(c,d));}template <class T>inline bool scan_d(T &ret){char c;int sgn;if (c = getchar(), c == EOF){return 0;}while (c != '-' && (c < '0' || c > '9')){c = getchar();}sgn = (c == '-') ? -1 : 1;ret = (c == '-') ? 0 : (c - '0');while (c = getchar(), c >= '0' && c <= '9'){ret = ret * 10 + (c - '0');}ret *= sgn;return 1;}inline bool scan_lf(double &num){char in;double Dec=0.1;bool IsN=false,IsD=false;in=getchar();if(in==EOF) return false;while(in!='-'&&in!='.'&&(in<'0'||in>'9'))in=getchar();if(in=='-'){IsN=true;num=0;}else if(in=='.'){IsD=true;num=0;}else num=in-'0';if(!IsD){while(in=getchar(),in>='0'&&in<='9'){num*=10;num+=in-'0';}}if(in!='.'){if(IsN) num=-num;return true;}else{while(in=getchar(),in>='0'&&in<='9'){num+=Dec*(in-'0');Dec*=0.1;}}if(IsN) num=-num;return true;}void Out(LL a){if(a < 0) { putchar('-'); a = -a; }if(a >= 10) Out(a / 10);putchar(a % 10 + '0');}void print(LL a){ Out(a),puts("");}//freopen( "in.txt" , "r" , stdin );//freopen( "data.txt" , "w" , stdout );//cerr << "run time is " << clock() << endl;int root,last;int tots; //总结点int l; //字符串长度struct sam_node{ int fa,son[26]; int len; void init(int _len) { len = _len; fa = -1; memset(son,-1,sizeof(son)); }}t[maxn*2];void sam_init(){ tots = 0; root = last = 0; t[tots].init(0);}void extend(char ch){ int w=ch-'a'; int p=last; int np=++tots;t[tots].init(t[p].len+1); int q,nq; while(p!=-1&&t[p].son[w]==-1){t[p].son[w]=np;p=t[p].fa;} if (p==-1) t[np].fa=root; else { q=t[p].son[w]; if (t[p].len+1==t[q].len){t[np].fa=q;} else { nq=++tots;t[nq].init(0); t[nq]=t[q]; t[nq].len=t[p].len+1; t[q].fa=nq;t[np].fa=nq; while(p!=-1&&t[p].son[w]==q){t[p].son[w]=nq;p=t[p].fa;} } } last=np;}int w[maxn], r[maxn*2];void topsort(){ for(int i = 0; i <= l; ++i) w[i] = 0; for(int i = 1; i <= tots; ++i) w[t[i].len]++; for(int i = 1; i <= l; ++i) w[i] += w[i-1]; for(int i = tots; i >= 1; --i) r[w[t[i].len]--] = i; r[0] = 0;}int dp[maxn*2];char s[maxn];int T;int main(){ scan_d(T); W(T--) { scanf("%s",s); l=strlen(s); sam_init(); FOr(0,l,i) extend(s[i]); FOr(0,l,i) extend(s[i]); int now=root; FOR(1,l,i) { FOr(0,26,j) { if(t[now].son[j]!=-1) { now=t[now].son[j]; break; } } } print(t[now].len-l+1); }}
阅读全文
0 0
- poj1509 Glass Beads 找最小的字母边转移
- 【POJ1509】Glass Beads 最小表示法
- POJ1509--Glass Beads
- poj1509 Glass Beads SAM
- poj1509 Glass Beads sam
- [SAM] POJ1509 Glass Beads
- POJ1509——Glass Beads(字符串的最小表示法)
- POJ1509:Glass Beads(循环同构最小表示)
- poj1509 Glass Beads,后缀自动机
- [POJ1509]Glass Beads && 后缀自动机
- poj1509 Glass Beads 后缀数组
- poj1509:Glass Beads(后缀自动机)
- ZJU2006 Glass Beads - 字符串的最小表示
- Glass Beads-最小表示法
- 最小表示法 模板【poj1509】Glass Bead
- POJ 1509 Glass Beads(字符串的最小表示法)
- [最小表示] poj 1509 Glass Beads
- UVa719 - Glass Beads(最小表示法)
- 继承AppCompatActivity的Activity无法隐藏标题栏
- android surfaceflinger(1)-启动初始化1
- 百度大牛讲机制设计和计算广告学
- 如何为treewidgetitem添加右键菜单
- 数组
- poj1509 Glass Beads 找最小的字母边转移
- OpenCV2编程手册笔记之 6.4高通滤波器检测边缘
- svn简介及使用
- python中if语句的真假判断
- 安卓按键命令库教程(紫猫版)
- js完成页面表格的增删
- 推荐系统基础知识
- Python 正则表达式匹配ip格式详解
- Pat(A) 1106. Lowest Price in Supply Chain (25)