zoukankan      html  css  js  c++  java
  • [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 }
  • 相关阅读:
    TimeSpan的操作
    List<T>的排序和筛选
    编程中的一些概念
    SVN返回版本
    语音播报
    优化编译器的局限性
    Inline Functions 与它的形式参数和局部变量
    函数的效能 & 指向 Member Functions 的指针与其效能
    虚拟继承下 Virtual Function 的语意
    多重继承下 Virtual Function 的语意
  • 原文地址:https://www.cnblogs.com/skylee03/p/7526748.html
Copyright © 2011-2022 走看看