题意:给你一堆字符串,仅包含数字'0'到'9'。

例如 101 123

有一个字符串集合S包含输入的N个字符串,和他们的全部字串。

操作字符串很无聊,你决定把它们转化成数字。

你可以把一个字符串转换成一个十进制整数。

如果一个数字出现了多次,只留一个。

计算所有数字的和,模2012。

1<=N<=10000

题解:参考 http://blog.csdn.net/kqzxcmh/article/details/8122747

每个后缀的所有前缀就是所有子串。sa[i]和sa[i-1]有height[i]的长度是重复的,可以不考虑。

把字符串连起来,中间加分隔符,构成一个大的字符串,然后整体求后缀数组。

对于一个字符串123
可以v[0]=1 v[1]=12 v[3]=123  v[i]是0-i构成的数字
sum是前缀和 sum[0]=1 sum[1]=13 sum[2]=136

对于后缀23 sum[2]-sum[0] 得到 12+123的和

对于12需要减去10 对于123需要减去100

所以求l~r 首先需要sum[r]-sum[l-1], 然后设x=r-l

需要减去 for(int i=1;i<=x;++i) pow(10, i);

可以预处理出来。

太难辣!!像我这种辣鸡不适合这么难的题T^T

时限3s,187ms还是比较快的。

//后缀数组
#include <stdio.h>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
typedef long long ll; const int N = int(2e5)+;
const int M = ; int cmp(int *r,int a,int b,int l){
return (r[a]==r[b]) && (r[a+l]==r[b+l]);
}
// 用于比较第一关键字与第二关键字,
// 比较特殊的地方是,预处理的时候,r[n]=0(小于前面出现过的字符) int wa[N],wb[N],wss[N],wv[N];
int sa[N]; // 排第i的是谁 i:1~n sa:0~n-1
int rk[N], // i排第几 i:0~n-1 rk:1~n
height[N]; // 排名相邻的两个后缀的最长公共前缀长度:suffix(sa[i-1])和(sa[i]) 的最长公共前缀,
char str[N]; void DA(char *r,int *sa,int n,int m){ // 此处N比输入的N要多1,为人工添加的一个字符,用于避免CMP时越界 m是不同的字符的个数
int i,j,p,*x=wa,*y=wb,*t;
for(i=;i<m;i++) wss[i]=;
for(i=;i<n;i++) wss[x[i]=r[i]]++;
for(i=;i<m;i++) wss[i]+=wss[i-];
for(i=n-;i>=;i--) sa[--wss[x[i]]]=i;
for(j=,p=;p<n;j*=,m=p)
{
for(p=,i=n-j;i<n;i++) y[p++]=i;
for(i=;i<n;i++) if(sa[i]>=j) y[p++]=sa[i]-j;
for(i=;i<n;i++) wv[i]=x[y[i]];
for(i=;i<m;i++) wss[i]=;
for(i=;i<n;i++) wss[wv[i]]++;
for(i=;i<m;i++) wss[i]+=wss[i-];
for(i=n-;i>=;i--) sa[--wss[wv[i]]]=y[i];
for(t=x,x=y,y=t,p=,x[sa[]]=,i=;i<n;i++)
x[sa[i]]=cmp(y,sa[i-],sa[i],j)?p-:p++;
}
} void calheight(char *r,int *sa,int n){ // 此处N为实际长度
int i,j,k=;
for(i=;i<=n;i++) rk[sa[i]]=i;
for(i=;i<n; height[rk[i++]] = k )
for(k?k--:,j=sa[rk[i]-]; r[i+k]==r[j+k]; k++);
} char word[N];
int d[N]; // 该字符串的结束位置
int sum[N];
int v[N];
int Pow[N]; int cal(int l, int r) {
if (l > r) return ;
//printf(">>%d %d %d %d %d\n", l, r, sum[r+1], sum[l], v[l]);
//printf("%d\n", ((sum[r+1]-sum[l]) - v[l]*Pow[r-l+1]%M + M) % M);
return ((sum[r+]-sum[l]) - v[l]*Pow[r-l+]%M + M) % M;
} int main(int argc, char const *argv[])
{
//freopen("in.txt", "r", stdin);
Pow[]=;Pow[]=;
for (int i=;i<N;++i)
Pow[i]=(Pow[i-]+)*%M;
int n;
while (~scanf("%d", &n)) {
int idx = ;
for (int i = ; i < n; ++i) {
scanf("%s", word);
int len = strlen(word);
int val = ;
for (int j = ; j < len; ++j) {
str[idx] = word[j] - '' + ;
val = (val * + str[idx] - ) % M;
v[idx+] = val; //+1是因为会有-1 <0很麻烦= =
sum[idx+] = (sum[idx] + val) % M;
d[idx] = len - j + idx;
idx++;
}
d[idx] = idx;
sum[idx+] = v[idx+] = ;
str[idx++] = ; //1~11
}
str[idx] = ;
DA(str, sa, idx+, );
calheight(str, sa, idx);
int ans = ;
for (int i = ; i < idx; ++i) {
if (str[i] == || str[i] == ) continue; // 不要前导零
int l = i + height[rk[i]];//重复的部分
int r = d[i]-;
//printf("i=%d,l=%d, r=%d\n", i, l, r);
if (l > r) continue;
ans = (ans + cal(i, r) - cal(i, l-)) % M;
//ans = (ans + cal(l, r)) % M; 这个不对 因为所求字符串是从i开始的
ans = (ans + M) % M;
}
printf("%d\n", ans);
}
return ;
}

后缀自动机= = 先留坑吧orz。。。。

hdu4436-str2int(后缀数组 or 后缀自动机)的更多相关文章

  1. [TJOI2015]弦论(后缀数组or后缀自动机)

    解法一:后缀数组 听说后缀数组解第k小本质不同的子串是一个经典问题. 把后缀排好序后第i个串的本质不同的串的贡献就是\(n-sa[i]+1-LCP(i,i-1)\)然后我们累加这个贡献,看到哪一个串的 ...

  2. (持续更新)虚树,KD-Tree,长链剖分,后缀数组,后缀自动机

    真的就是讲课两天,吸收一个月呢! \(1.\)虚树 \(2.\)KD-Tree \(3.\)长链剖分 \(4.\)后缀数组 后缀数组 \(5.\)后缀自动机 后缀自动机

  3. 字符串数据结构模板/题单(后缀数组,后缀自动机,LCP,后缀平衡树,回文自动机)

    模板 后缀数组 #include<bits/stdc++.h> #define R register int using namespace std; const int N=1e6+9; ...

  4. poj 2774 最长公共子--弦hash或后缀数组或后缀自己主动机

    http://poj.org/problem?id=2774 我想看看这里的后缀数组:http://blog.csdn.net/u011026968/article/details/22801015 ...

  5. poj2774 Long Long Message(后缀数组or后缀自动机)

    转载请注明出处: http://www.cnblogs.com/fraud/          ——by fraud Long Long Message Time Limit: 4000MS   Me ...

  6. 字符串 --- KMP Eentend-Kmp 自动机 trie图 trie树 后缀树 后缀数组

    涉及到字符串的问题,无外乎这样一些算法和数据结构:自动机 KMP算法 Extend-KMP 后缀树 后缀数组 trie树 trie图及其应用.当然这些都是比较高级的数据结构和算法,而这里面最常用和最熟 ...

  7. bzoj 3172 后缀数组|AC自动机

    后缀数组或者AC自动机都可以,模板题. /************************************************************** Problem: 3172 Us ...

  8. SPOJ694 DISUBSTR --- 后缀数组 / 后缀自动机

    SPOJ694 DISUBSTR 题目描述: Given a string, we need to find the total number of its distinct substrings. ...

  9. POJ3080 POJ3450Corporate Identity(广义后缀自动机||后缀数组||KMP)

    Beside other services, ACM helps companies to clearly state their “corporate identity”, which includ ...

随机推荐

  1. 1989-C. 数字三角形

    描述 如图所示,是一个数字搭成的三角形. 若起始位置在三角形的顶端,结束位置在三角形底边,每一步只能向下方或向右下角移动一格.请编程计算一条路径,使得路径上经过的数字和最大.(图中路径7→3→8→7→ ...

  2. HDU4539+状态压缩DP

    /* 题意:n行m列的矩阵,1表示可以放东西,0表示不可以.曼哈顿距离为2的两个位置最多只能有一个位置放东西. 问最多放多少个东西. */ #include<stdio.h> #inclu ...

  3. valgrind基本使用

    1.valgrind是一个内存检测工具,类似的还有purify,insure++等 2.测试文件test.c test.c : main(){ int* a=new int[100]; return ...

  4. 两个基于C++/Qt的开源WEB框架

    1.tufao 项目地址: https://github.com/vinipsmaker/tufao 主页: http://vinipsmaker.github.io/tufao/ 介绍: Tufão ...

  5. Mac查看端口占用情况

    Mac下使用lsof(list open files)来查看端口占用情况,lsof 是一个列出当前系统打开文件的工具. 使用 lsof 会列举所有占用的端口列表: $ lsof 使用less可以用于分 ...

  6. linux shell 常用基本语法

    转自网络,真正来源不详.... 一. Linux基本命令 1.1.  cp命令 该命令的功能是将给出的文件或目录拷贝到另一文件或目录中,功能十分强大. 语法: cp [选项] 源文件或目录 目标文件或 ...

  7. 彻底搞清js中闭包(Closure)的概念

    js中闭包这个概念对于初学js的同学来说, 会比较陌生, 有些难以理解, 理解起来非常模糊. 今天就和大家一起来探讨一下这个玩意. 相信大家在看完后, 心中的迷惑会迎然而解. 闭包概念: 闭包就是有权 ...

  8. bzoj2595

    一开始看是插头dp,后来发现还有一个叫斯坦纳树的东西 什么叫斯坦纳树,就是使给定点连通开销和最小的树(可以包含多余的点) 到这张平面图上,我们不难想到用dp来解决,设f[x,y,S]表示连通集合为S, ...

  9. poj2761

    表面上看是主席树之类的区间k大 实际上,除了主席树,还可以测各种结构 因为题目中说,任意区间不会完全包含 于是,我们把区间按左端点排序,依次添加,用平衡树求当前的k大 每个狗最多被添加一次,删除一次 ...

  10. 对于requirejs AMD模块加载的理解

    个人人为使用模块化加载的优点有三: 1,以我的项目为例:90%为图表展示,使用的是echarts,此文件较大,requirejs可以在同一个版本号(urlArgs)之下缓存文件,那么我就可以在访问登陆 ...