[SPOJ-BEADS]Glass Beads

 

来源:
  CE1998

题目大意:
  求字符串最小表示。

思路:
  字符串复制一遍接在后面,构建SAM,然后每次跑小的转移。
  跑n次以后就跑到了最小表示的末尾,用该状态的len值减去n就是最小表示的起始位置。

 1 #include<string>
 2 #include<iostream>
 3 int n;
 4 std::string s;
 5 class SuffixAutomaton {
 6     private:
 7         static const int SIGMA_SIZE=26;
 8         struct State {
 9             State *link,*trans[SIGMA_SIZE];
10             unsigned len,min_trans;
11             State(const int l) {
12                 link=nullptr;
13                 std::fill(&trans[0],&trans[SIGMA_SIZE],nullptr);
14                 min_trans=SIGMA_SIZE;
15                 len=l;
16             }
17         };
18         State *root,*last;
19         unsigned idx(const int ch) {
20             return ch-'a';
21         }
22         void extend(const char ch) {
23             const unsigned w=idx(ch);
24             State *p=last,*new_p=new State(last->len+1);
25             while(p!=nullptr&&p->trans[w]==nullptr) {
26                 p->trans[w]=new_p;
27                 p->min_trans=std::min(p->min_trans,w);
28                 p=p->link;
29             }
30             if(p==nullptr) {
31                 new_p->link=root;
32             } else {
33                 State *q=p->trans[w];
34                 if(q->len==p->len+1) {
35                     new_p->link=q;
36                 } else {
37                     State *new_q=new State(p->len+1);
38                     std::copy(&q->trans[0],&q->trans[SIGMA_SIZE],new_q->trans);
39                     new_q->min_trans=q->min_trans;
40                     new_q->link=q->link;
41                     q->link=new_p->link=new_q;
42                     while(p!=nullptr&&p->trans[w]==q) {
43                         p->trans[w]=new_q;
44                         p=p->link;
45                     }
46                 }
47             }
48             last=new_p;
49         }
50     public:
51         void build() {
52             root=last=new State(0);
53             for(std::string::iterator i=s.begin();i<s.end();i++) {
54                 extend(*i);
55             }
56             for(std::string::iterator i=s.begin();i<s.end();i++) {
57                 extend(*i);
58             }
59         }
60         unsigned query() {
61             State *p=root;
62             for(unsigned i=0;i<s.length();i++) {
63                 p=p->trans[p->min_trans];
64             }
65             return p->len-s.length()+1;
66         }
67 };
68 SuffixAutomaton sam;
69 int main() {
70     std::ios_base::sync_with_stdio(false);
71     std::cin.tie(NULL);
72     std::cin>>n;
73     for(int i=0;i<n;i++) {
74         std::cin>>s;
75         sam.build();
76         std::cout<<sam.query()<<std::endl; 
77     }
78     return 0;
79 }

 

posted @ 2017-09-15 15:38  skylee03  阅读(136)  评论(0编辑  收藏  举报