题目链接:https://cn.vjudge.net/contest/283743#problem/F

题目大意:给你一个字符串,然后让你求出不同的子串的个数。

具体思路:首先,一个字符串中总的子串个数是len*(len+1)/2,然后就开始去重了,通过height数组,求出所有重复的子串的个数,然后就用总的子串的个数-重复的子串的个数就可以了。

AC代码:

  1. #include<iostream>
  2. #include<stack>
  3. #include<cstring>
  4. #include<iomanip>
  5. #include<stdio.h>
  6. #include<algorithm>
  7. #include<cmath>
  8. using namespace std;
  9. # define ll long long
  10. const int maxn = 5e5+;
  11. int cntA[maxn], cntB[maxn], sa[maxn], tsa[maxn], A[maxn], B[maxn], height[maxn];
  12. int Rank[maxn];
  13. char ch[maxn];
  14. ll n;
  15. //sa[i]代表第i小的后缀位置,Rank[i]代表第i位置开始的后缀在所有的后缀串中排名第几
  16. // height[i]代表排名第i个字符串和第i-1个字符串的相同前缀的长度
  17. void cal()
  18. {
  19. for(int i = ; i < ; i++) cntA[i] = ;
  20. for(int i = ; i <= n; i++) cntA[ch[i-]]++;
  21. for(int i = ; i < ; i++) cntA[i] += cntA[i-];
  22. for(int i = n; i; i--) sa[cntA[ch[i-]]--] = i;
  23. Rank[sa[]] = ;
  24. for(int i = ; i <= n; i++)
  25. {
  26. Rank[sa[i]] = Rank[sa[i-]];
  27. if(ch[sa[i]-] != ch[sa[i-]-]) Rank[sa[i]]++;
  28. }
  29. for(int l = ; Rank[sa[n]] < n; l <<= )
  30. {
  31. memset(cntA, , sizeof(cntA));
  32. memset(cntB, , sizeof(cntB));
  33. for(int i = ; i <= n; i++)
  34. {
  35. cntA[A[i] = Rank[i]]++;
  36. cntB[B[i] = (i+l <= n)?Rank[i+l]:]++;
  37. }
  38. for(int i = ; i <= n; i++) cntB[i] += cntB[i-];
  39. for(int i = n; i; i--) tsa[cntB[B[i]]--] = i;
  40. for(int i = ; i <= n; i++) cntA[i] += cntA[i-];
  41. for(int i = n; i; i--) sa[cntA[A[tsa[i]]]--] = tsa[i];
  42. Rank[sa[]]=;
  43. for(int i = ; i <= n; i++)
  44. {
  45. Rank[sa[i]] = Rank[sa[i-]];
  46. if(A[sa[i]] != A[sa[i-]] || B[sa[i]] != B[sa[i-]]) Rank[sa[i]]++;
  47. }
  48. }
  49. for(int i = , j = ; i <= n; i++)
  50. {
  51. if(j) j--;
  52. while(ch[i+j-] == ch[sa[Rank[i]-] + j - ]) j++;
  53. height[Rank[i]] = j;
  54. }
  55. }
  56. int main()
  57. {
  58. int T;
  59. scanf("%d",&T);
  60. while(T--)
  61. {
  62. scanf("%s",ch);
  63. n=strlen(ch);
  64. if(n==)
  65. {
  66. printf("1\n");
  67. continue;
  68. }
  69. // cout<<1<<endl;
  70. cal();
  71. ll ans=n*(n+)/;
  72. //cout<<ans<<endl;
  73. for(int i=; i<=n; i++)
  74. {
  75. ans-=height[i];
  76. // cout<<i<<" "<<height[i]<<endl;
  77. }
  78. printf("%lld\n",ans);
  79. }
  80. return ;
  81. }

F - New Distinct Substrings (后缀数组)的更多相关文章

  1. SPOJ - SUBST1 New Distinct Substrings —— 后缀数组 单个字符串的子串个数

    题目链接:https://vjudge.net/problem/SPOJ-SUBST1 SUBST1 - New Distinct Substrings #suffix-array-8 Given a ...

  2. SPOJ - DISUBSTR Distinct Substrings (后缀数组)

    Given a string, we need to find the total number of its distinct substrings. Input T- number of test ...

  3. 【SPOJ – SUBST1】New Distinct Substrings 后缀数组

    New Distinct Substrings 题意 给出T个字符串,问每个字符串有多少个不同的子串. 思路 字符串所有子串,可以看做由所有后缀的前缀组成. 按照后缀排序,遍历后缀,每次新增的前缀就是 ...

  4. SPOJ DISUBSTR Distinct Substrings 后缀数组

    题意:统计母串中包含多少不同的子串 然后这是09年论文<后缀数组——处理字符串的有力工具>中有介绍 公式如下: 原理就是加上新的,减去重的,这题是因为打多校才补的,只能说我是个垃圾 #in ...

  5. SPOJ 694 || 705 Distinct Substrings ( 后缀数组 && 不同子串的个数 )

    题意 : 对于给出的串,输出其不同长度的子串的种类数 分析 : 有一个事实就是每一个子串必定是某一个后缀的前缀,换句话说就是每一个后缀的的每一个前缀都代表着一个子串,那么如何在这么多子串or后缀的前缀 ...

  6. spoj Distinct Substrings 后缀数组

    给定一个字符串,求不相同的子串的个数. 假如给字符串“ABA";排列的子串可能: A B A AB  BA ABA 共3*(3+1)/2=6种; 后缀数组表示时: A ABA BA 对于A和 ...

  7. [spoj694&spoj705]New Distinct Substrings(后缀数组)

    题意:求字符串中不同子串的个数. 解题关键:每个子串一定是某个后缀的前缀,那么原问题等价于求所有后缀之间的不相同的前缀的个数. 1.总数减去height数组的和即可. 注意这里height中为什么不需 ...

  8. spoj 694. Distinct Substrings 后缀数组求不同子串的个数

    题目链接:http://www.spoj.com/problems/DISUBSTR/ 思路: 每个子串一定是某个后缀的前缀,那么原问题等价于求所有后缀之间的不相同的前缀的个数.如果所有的后缀按照su ...

  9. SPOJ_705_New Distinct Substrings_后缀数组

    SPOJ_705_New Distinct Substrings_后缀数组 题意: 给定一个字符串,求该字符串含有的本质不同的子串数量. 后缀数组的一个小应用. 考虑每个后缀的贡献,如果不要求本质不同 ...

  10. SPOJ- Distinct Substrings(后缀数组&后缀自动机)

    Given a string, we need to find the total number of its distinct substrings. Input T- number of test ...

随机推荐

  1. AtCoder WTF 2019 C2. Triangular Lamps Hard

    题目链接 感觉这样的题真的称得上是鬼斧神工啊,\(\text{OI}\)中能多一些这样的题目就太好了. 题意: 有一个二维的三角坐标系,大概如图所示(图是从atcoder里偷下来的): 坐标系上的每个 ...

  2. docker--Dockerfile--sonarqube

    FROM openjdk:8 ENV SONAR_VERSION=6.7.1 \ SONARQUBE_HOME=/opt/sonarqube \ # Database configuration # ...

  3. BZOJ[Usaco2017 Jan]Promotion Counting——线段树合并

    题目描述 The cows have once again tried to form a startup company, failing to remember from past experie ...

  4. ubuntu16.04 NFS系统挂载

    一:服务器端 step1:关闭防火墙 sudo ufw disable step2:安装nfs sudo apt-get install nfs-kernel-server step3: 打开/etc ...

  5. Codeforces Round #337 (Div. 2) B. Vika and Squares

    B. Vika and Squares time limit per test 2 seconds memory limit per test 256 megabytes input standard ...

  6. 通过 powershell 配置 IIS

    1. 设置iis pool: cls Import-Module WebAdministration Get-ChildItem IIS:\apppools | ForEach-Object{     ...

  7. linux 分区、目录及用途

    主要分区: 目录 建议大小 格式 描述 / 10G-20G ext4 根目录 swap <2048M swap 交换空间 /boot 200M左右 ext4 Linux的内核及引导系统程序所需要 ...

  8. 快速幂&快速乘法

    尽管快速幂与快速乘法好像扯不上什么关系,但是东西不是很多,就一起整理到这里吧 快速幂思想就是将ax看作x个a相乘,用now记录当前答案,然后将指数每次除以2,然后将当前答案平方,如果x的2进制最后一位 ...

  9. 获取天气预报API5_统计最容易生病时间段

    sklearn实战-乳腺癌细胞数据挖掘(博客主亲自录制视频教程) https://study.163.com/course/introduction.htm?courseId=1005269003&a ...

  10. linux tail -f messages查看控制台失败

    [root@localhost log]# tail -f /var/log/messages ......................... tail: cannot watch `/var/l ...