0、题意:给定你n个字符串,询问每个字符串有多少子串(不包括空串)是所有n个字符串中至少k个字符串的子串(注意包括本身)。

1、分析:这个题我问了吴大爷做法

首先建立后缀自动机,然后利用离线搞出每一个节点在多少个串中,这个用树状数组统计一下就好,和BZOJ2780一样,然后如果这个节点在不少于x个串中,我们把这个点的value赋值成这个节点父亲边的字符个数,否则为0。随后我们预处理一下从根到每个节点路径上的权值和。于是每个字符串的答案等于所有这个字符串的后缀节点的从根到该节点的权值和。

然后这个题就解决了

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
using namespace std;
#define M 200010
#define LL long long

inline int read(){
    char ch = getchar(); int x = 0, f = 1;
    while(ch < '0' || ch > '9'){
        if(ch == '-') f = -1;
        ch = getchar();
    }
    while('0' <= ch && ch <= '9'){
        x = x * 10 + ch - '0';
        ch = getchar();
    }
    return x * f;
}

struct node{
    int tranc[27], fa, len;
} a[202010];
int cnt, tail;
char s[202010];
int p, nq, q, np;
int in[M], out[M], seg[M], num;
int C[M], v[M];
int lastins[M];
int sum[M];

struct gragh{
    int head[M], to[M], Next[M], tot;

    inline void init(){
        memset(head, -1, sizeof(head));
        tot = 0;
    }

    inline void add(int u, int v){
        ++ tot;
        Next[tot] = head[u];
        head[u] = tot;
        to[tot] = v;
    }
} w, G, sam;

struct Query{
    int id, l, r;
    inline bool operator < (const Query& rhs) const{
        return r < rhs.r;
    }
} S[M];

inline void insert(int k, int op){
    p = tail;
    if(a[p].tranc[k] != 0){
        q = a[p].tranc[k];
        if(a[q].len == a[p].len + 1) tail = q;
        else{
            a[nq = ++ cnt] = a[q];
            a[q].fa = nq;
            a[nq].len = a[p].len + 1;
            for(; a[p].tranc[k] == q; p = a[p].fa) a[p].tranc[k] = nq;
            tail = nq;
        }
    }
    else{
        np = ++ cnt;
        a[np].len = a[tail].len + 1;
        for(p = tail; p && !a[p].tranc[k]; p = a[p].fa) a[p].tranc[k] = np;
        if(!p) a[np].fa = 1;
        else{
            q = a[p].tranc[k];
            if(a[q].len == a[p].len + 1) a[np].fa = q;
            else{
                a[nq = ++ cnt] = a[q];
                a[q].fa = a[np].fa = nq;
                a[nq].len = a[p].len + 1;
                for(; a[p].tranc[k] == q; p = a[p].fa) a[p].tranc[k] = nq;
            }
        }
        tail = np;
    }
    w.add(tail, op);
    sam.add(op, tail);
}

inline void dfs(int x){
    in[x] = ++ num; seg[num] = x;
    for(int i = G.head[x]; i != -1; i = G.Next[i]){
        dfs(G.to[i]);
    }
    out[x] = num;
}

inline void change(int x, int d){
    for(; x <= cnt; x += (x & -x)) C[x] += d;
}

inline int ask(int x){
    int ret = 0;
    for(; x > 0; x -= (x & -x)) ret += C[x];
    return ret;
}

inline void dfs2(int x){
    v[x] += v[a[x].fa];
    for(int i = G.head[x]; i != -1; i = G.Next[i]) dfs2(G.to[i]);
}

int main(){
    int n = read(), lim = read();
    cnt = 1;
    G.init(); w.init(); sam.init();
    for(int i = 1; i <= n; i ++){
        scanf("%s", s + 1);
        tail = 1;
        int len = strlen(s + 1);
        for(int j = 1; j <= len; j ++){
            insert(s[j] - 'a' + 1, i);
        }
    }
    for(int i = 1; i <= cnt; i ++){
        if(a[i].fa != 0) G.add(a[i].fa, i);
    }
    dfs(1);
    for(int i = 1; i <= cnt; i ++) S[i] = (Query){i, in[i], out[i]};

    sort(S + 1, S + cnt + 1);
    int k = 1;
    for(int i = 1; i <= cnt; i ++){
        for(int j = w.head[seg[i]]; j != -1; j = w.Next[j]){
            if(lastins[w.to[j]]) change(lastins[w.to[j]], -1);
            change(lastins[w.to[j]] = i, 1);
        }
        while(S[k].r == i) sum[S[k].id] = ask(S[k].r) - ask(S[k].l - 1), k ++;
    }
    for(int i = 1; i <= cnt; i ++){
        if(sum[i] >= lim) v[i] = a[i].len - a[a[i].fa].len;
        else v[i] = 0;
    }
    dfs2(1);
    LL ans = 0;
    for(int i = 1; i <= n; i ++){
        if(i > 1) printf(" ");
        ans = 0;
        for(int j = sam.head[i]; j != -1; j = sam.Next[j]) ans += v[sam.to[j]];
        printf("%lld", ans);
    }
    return 0;
}

BZOJ3277——串的更多相关文章

  1. BZOJ3473&&BZOJ3277串

    BZOJ3473&&BZOJ3277串 题面 自己找去 HINT 对于所有串建立一个广义后缀自动机,对于每一个节点开一个set表示这个节点接受的子串在哪些串里出现过,然后在parent ...

  2. BZOJ3277 串(后缀数组+二分答案+主席树)

    因为不会SAM,考虑SA.将所有串连起来并加分隔符,每次考虑计算以某个位置开始的子串有多少个合法. 对此首先二分答案,找到名次数组上的一个区间,那么只需要统计有多少个所给串在该区间内出现就可以了.这是 ...

  3. bzoj3277 串 (后缀数组+二分答案+ST表)

    常见操作:先把所有串都连到一起,但中间加上一个特殊的符号(不能在原串中/出现过)作为分割 由于全部的子串就等于所有后缀的所有前缀,那我们对于每一个后缀,去求一个最长的前缀,来满足这个前缀在至少K个原串 ...

  4. bzoj3473: 字符串 && bzoj3277串

    3473: 字符串 Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 121  Solved: 53[Submit][Status][Discuss] D ...

  5. 【文文殿下】[BZOJ3277] 串

    Description 字符串是oi界常考的问题.现在给定你n个字符串,询问每个字符串有多少子串(不包括空串)是所有n个字符串中 至少k个字符串的子串(注意包括本身) Input 第一行两个整数n,k ...

  6. BZOJ3277 串 【广义后缀自动机】

    Description 字符串是oi界常考的问题.现在给定你n个字符串,询问每个字符串有多少子串(不包括空串)是所有n个字符串中 至少k个字符串的子串(注意包括本身). Input 第一行两个整数n, ...

  7. bzoj3473字符串&bzoj3277串

    题意:给定n个字符串,询问每个字符串有多少子串(不包括空串)是所有n个字符串中至少k个字符串的子串.注意本质相同的子串多次出现算多次,如1 1 aaa这组数据答案为6,贡献1WA.代码里有些部分是为了 ...

  8. BZOJ3277: 串(广义后缀自动机)

    Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1196  Solved: 478[Submit][Status][Discuss] Descripti ...

  9. BZOJ3277 串(后缀自动机)

    对多串建立SAM的一种方法是加分隔符.于是加完分隔符建出SAM. 考虑统计出每个节点被多少个串包含.让每个串各自在SAM上跑,跑到一个节点就标记(显然一定会完全匹配该节点,因为是对包含其的串建的SAM ...

随机推荐

  1. jQuery实现表格行的动态增加与删除 序号 从 1开始排列

    <table id="tab" border="1" width="60%" align="center" sty ...

  2. 用sql 语句给字段添加描述

    用sql 语句给字段添加描述 IF not exists (SELECT * FROM ::fn_listextendedproperty (NULL, 'user', 'dbo', 'table', ...

  3. 兼容ie6及一下版本的自适应

    <meta http-equiv="X-UA-Compatible" content="IE=edge,Chrome=1" /> <meta ...

  4. oracle表字段为汉字,依据拼音排序

    在order by后面使用NLSSORT函数转化汉字列,如下 select * from student order by NLSSORT(name,'NLS_SORT=SCHINESE_PINYIN ...

  5. JavaWeb学习笔记——访问静态HTML网页

    基础知识: 1.一个标准 的WEB应用,在根目录下必须有一个WEB-INF文件夹,文件夹下必须有一个WEB.xml文件. 2.classes文件夹用来存放编译的class文件. 3.lib文件夹用来存 ...

  6. Unity 下载

    Unity历史版本 http://wiki.ceeger.com/unity:history#unity_522f1 UNITY 下载存档 http://unity3d.com/cn/get-unit ...

  7. git命令之git tag 给当前分支打标签

    git tag - 标签相关操作 发表于 2011年06月29日 由 机器猫 标签可以针对某一时间点的版本做标记,常用于版本发布. 列出标签 $ git tag # 在控制台打印出当前仓库的所有标签$ ...

  8. php 数组排序以及按照某个字段排序

    经常,开发人员发现在PHP中使用这种数据结构对值或者数组元素进行排序非常有用.PHP提供了一些适合多种数组的排序函数,这些函数允许你在数组内部对元素进行排列,也允许用很多不同的方法对它们进行重新排序. ...

  9. 【好玩】将js代码转为日式表情

    直接上网址:http://utf-8.jp/public/aaencode.html?src=hello 试试将下面代码执行一下吧: ゚ω゚ノ= /`m´)ノ ~┻━┻ //*´∇`*/ ['_']; ...

  10. zookeeper运维 --【】转】

     from:http://blog.csdn.net/hengyunabc/article/details/19006911  zookeeper运维 尽管zookeeper在编程上有很多的阱陷,AP ...