zoukankan      html  css  js  c++  java
  • BZOJ 3926 && ZJOI 2015 诸神眷顾的幻想乡 (广义后缀自动机)

    3926: [Zjoi2015]诸神眷顾的幻想乡
    
    Time Limit: 10 Sec  Memory Limit: 512 MB
    
    Description
    幽香是全幻想乡里最受人欢迎的萌妹子,这天,是幽香的2600岁生日,无数幽香的粉丝到了幽香家门前的太阳花田上来为幽香庆祝生日。 
    粉丝们非常热情,自发组织表演了一系列节目给幽香看。幽香当然也非常高兴啦。 
    这时幽香发现了一件非常有趣的事情,太阳花田有n块空地。在过去,幽香为了方便,在这n块空地之间修建了n-1条边将它们连通起来。也就是说,这n块空地形成了一个树的结构。 
    有n个粉丝们来到了太阳花田上。为了表达对幽香生日的祝贺,他们选择了c中颜色的衣服,每种颜色恰好可以用一个0到c-1之间的整数来表示。并且每个人都站在一个空地上,每个空地上也只有一个人。这样整个太阳花田就花花绿绿了。幽香看到了,感觉也非常开心。 
    粉丝们策划的一个节目是这样的,选中两个粉丝A和B(A和B可以相同),然后A所在的空地到B所在的空地的路径上的粉丝依次跳起来(包括端点),幽香就能看到一个长度为A到B之间路径上的所有粉丝的数目(包括A和B)的颜色序列。一开始大家打算让人一两个粉丝(注意:A,B和B,A是不同的,他们形成的序列刚好相反,比如红绿蓝和蓝绿红)都来一次,但是有人指出这样可能会出现一些一模一样的颜色序列,会导致审美劳。 
    于是他们想要问题,在这个树上,一共有多少可能的不同的颜色序列(子串)幽香可以看到呢? 
    太阳花田的结构比较特殊,只与一个空地相邻的空地数量不超过20个。 
    
    Input
    第一行两个正整数n,c。表示空地数量和颜色数量。 
    第二行有n个0到c-1之间,由空格隔开的整数,依次表示第i块空地上的粉丝的衣服颜色。(这里我们按照节点标号从小到大的顺序依次给出每块空地上粉丝的衣服颜色)。 
    接下来n-1行,每行两个正整数u,v,表示有一条连接空地u和空地v的边。 
    
    Output
    一行,输出一个整数,表示答案。 
    
    Sample Input
    7 3
    0 2 1 2 1 0 0 
    1 2
    3 4
    3 5
    4 6
    5 7
    2 5
    
    Sample Output
    30
    
    HINT
    对于所有数据,1<=n<=100000, 1<=c<=10。 
    对于15%的数据,n<=2000。 
    另有5%的数据,所有空地都至多与两个空地相邻。 
    另有5%的数据,除一块空地与三个空地相邻外,其他空地都分别至多与两个空地相邻。 
    另有5%的数据,除某两块空地与三个空地相邻外,其他空地都分别至多与两个空地相邻
    

    算法讨论:

    由于叶子节点不超过20个,所以我们分别从每个叶子出发建立广义后缀自动机。注意树上的自动建立要时刻清楚last节点是哪个。

    然后就可以发现,树的任何一个子串,都是自动机上的某个节点到其子孙的路径。然后本质不同的子串数量就是每个点和其pre指针的len差值。

    加和就可以啦~~~

    代码:

    #include <cstdlib>
    #include <cstring>
    #include <cstdio>
    #include <algorithm>
    #include <iostream>
     
    using namespace std;
    const int N = 100000 + 5;
    const int C = 10;
     
    struct State {
      int len, pre, next[C];
    }st[N * 40];
     
    struct SuffixAutomaton {
      int sz;
      void Init() {
        sz = 1;
        st[sz].pre = -1; st[sz].len = 0;
        sz ++;
      }
      int add(int c, int last) {
        int cur = sz ++, p;
        st[cur].len = st[last].len + 1;
        for(p = last; p != -1 && !st[p].next[c]; p = st[p].pre)
          st[p].next[c] = cur;
        if(p == -1) st[cur].pre = 1;
        else {
          int q = st[p].next[c];
          if(st[q].len == st[p].len + 1) st[cur].pre = q;
          else {
            int cle = sz ++;
            st[cle].pre = st[q].pre;
            st[cle].len = st[p].len + 1;
            for(int i = 0; i < C; ++ i) st[cle].next[i] = st[q].next[i];
            for(; p != -1 && st[p].next[c] == q; p = st[p].pre)
              st[p].next[c] = cle;
            st[q].pre = st[cur].pre = cle;
          }
        }
        last = cur;
        return last;
      }
      void solve() {
        long long ans = 0;
        for(int i = 2; i < sz; ++ i)
            ans = (long long) ans + st[i].len - st[st[i].pre].len;
        printf("%lld
    ", ans);
      }
    }sam;
     
    int n, c, cnt;
    int head[N << 1], w[N << 1], du[N];
    struct Edge {
      int from, to, next;
    }edges[N << 1];
     
    void insert(int from, int to) {
      ++ cnt;
      edges[cnt].from = from; edges[cnt].to = to;
      edges[cnt].next = head[from]; head[from] = cnt;
    }
     
    void dfs(int u, int f, int last) {
      last = sam.add(w[u], last);
      for(int i = head[u]; i; i = edges[i].next) {
        int v = edges[i].to;
        if(v != f) dfs(v, u, last);
      }
    }
     
    int main() {
      //freopen("zjoi15_substring.in", "r", stdin);
      //freopen("zjoi15_substring.out", "w", stdout);
       
      int x, y;
      scanf("%d%d", &n, &c);
      for(int i = 1; i <= n; ++ i) scanf("%d", &w[i]);
      for(int i = 1; i < n; ++ i) {
        scanf("%d%d", &x, &y);
        insert(x, y); insert(y, x);
        du[x] ++; du[y] ++;
      }
      sam.Init();
      for(int i = 1; i <= n; ++ i)
        if(du[i] == 1) { dfs(i, 0, 1); }
      sam.solve();
     
      //fclose(stdin); fclose(stdout);
      return 0;
    }
    
  • 相关阅读:
    ubuntu1804安装二进制版(tgz版编译版)mongdb4.2版的笔记
    ubuntu1804安装二进制版(编译版)nodejs12
    ubuntu1804安装pycharm2019.3.4版本及永久激活的方法
    ubuntu1804搜狗拼音输入法突然失效的解决办法
    2020ruby和ruby on rails想说再爱你不容易:安装rails失败解决办法
    ubuntu1804安装rubymine的IDE笔记
    ubuntu系统mysql5.7忘记/设置root的坑
    使用matplotlib时报which is a non-GUI backend的解决办法
    python文件操作笔记
    python字符串string的使用笔记
  • 原文地址:https://www.cnblogs.com/sxprovence/p/5396742.html
Copyright © 2011-2022 走看看