CodeForces 235C Cyclical Quest (后缀自动机)

题意:给一个主串,再给出多个模式串,分别求主串中有多少个连续子串,与模式串循环同构。

题解:后缀自动机
因为要求循环同构,所以将模式串复制放到后面。(要么加终止符,要么传入长度)
先对主串建sam,然后跑拓扑,自底向上更新 c n t cnt cnt t o p s a m [ ] topsam[] topsam[]),然后求 L C S LCS LCS
如果长度大于等于 l l l,跳 f a fa fa指针到长度为 l l l的节点 while (p->fa && p->fa->len >= l) p = p->fa;。那么此时状态所表示的子串个数 c n t cnt cnt即为所求,因为状态可能重复,用 v i s vis vis标记一下有没有被访问过即可 if (p->vis != id) p->vis = id, ans += p->cnt;
记得初始化 n o d e node node中新增变量。

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
#include<cstdio>
#include<string>
#include<cstring>
#include<algorithm>
#include<queue>
#include<stack>
#include<cmath>
#include<vector>
#include<fstream>
#include<set>
#include<map>
#define ll long long
using namespace std;
const int CHAR = 26;
const int MAXN = 1000010;
struct SAM_Node {
    SAM_Node* fa, * next[CHAR];
    int len;
    int id, pos;
    int vis, cnt;
    SAM_Node() {}
    SAM_Node(int _len) {
        fa = 0;
        len = _len;
        memset(next, 0, sizeof(next));
    }
};
SAM_Node SAM_node[MAXN * 2], * SAM_root, * SAM_last, * topsam[MAXN * 2];
int SAM_size, topcnt[MAXN * 2];
SAM_Node* newSAM_Node(int len) {
    SAM_node[SAM_size] = SAM_Node(len);
    SAM_node[SAM_size].id = SAM_size;
    SAM_node[SAM_size].cnt = SAM_node[SAM_size].vis = 0;
    return &SAM_node[SAM_size++];
}
SAM_Node* newSAM_Node(SAM_Node* p) {
    SAM_node[SAM_size] = *p;
    SAM_node[SAM_size].id = SAM_size;
    SAM_node[SAM_size].cnt = SAM_node[SAM_size].vis = 0;
    return &SAM_node[SAM_size++];
}
void SAM_init() {
    SAM_size = 0;
    SAM_root = SAM_last = newSAM_Node(0);
    SAM_node[0].pos = 0;
}
void SAM_add(int x, int len) {  //len从1开始
    SAM_Node* p = SAM_last, * np = newSAM_Node(p->len + 1);
    np->pos = len;
    SAM_last = np;
    for (; p && !p->next[x]; p = p->fa)
        p->next[x] = np;
    if (!p) {
        np->fa = SAM_root;
        return;
    }
    SAM_Node* q = p->next[x];
    if (q->len == p->len + 1) {
        np->fa = q;
        return;
    }
    SAM_Node* nq = newSAM_Node(q);
    nq->len = p->len + 1;
    q->fa = np->fa = nq;
    for (; p && p->next[x] == q; p = p->fa) p->next[x] = nq;
}
void SAM_build(char* s) {
    SAM_init();
    int len = strlen(s);
    for (int i = 0; i < len; i++) SAM_add(s[i] - 'a', i + 1);
    SAM_Node* p = SAM_root;
    for (int i = 0; i < len; i++) {
        p = p->next[s[i] - 'a'];
        p->cnt = 1;
    }
}
void topo() {
    memset(topcnt, 0, sizeof(topcnt));
    for (int i = 0; i <= SAM_size; i++) topcnt[SAM_node[i].len]++;
    for (int i = 1; i <= SAM_size; i++) topcnt[i] += topcnt[i - 1];
    for (int i = 0; i <= SAM_size; i++) topsam[--topcnt[SAM_node[i].len]] = &SAM_node[i];
}
void LCS(char* s, int id, int l) {
    int ans = 0, len = 0;
    SAM_Node* p = SAM_root;
    //cout << strlen(s) << endl;
    for (int i = 0; i < (l << 1) - 1; i++) {
        int idx = s[i] - 'a';
        if (p->next[idx]) p = p->next[idx], len++;
        else {
            while (p && !p->next[idx]) p = p->fa;
            if (!p) p = SAM_root, len = 0;
            else len = p->len + 1, p = p->next[idx];
        }
        //ans = max(ans, len);
        if (len >= l) {
            while (p->fa && p->fa->len >= l) p = p->fa;
            if (p->vis != id) p->vis = id, ans += p->cnt;
        }
    }
    printf("%d\n", ans);
}
char s[MAXN], t[MAXN];
int q;
int main() {
    scanf("%s%d", s, &q);
    SAM_build(s);
    topo();
    for (int i = SAM_size; i >= 2; i--) topsam[i]->fa->cnt += topsam[i]->cnt;
    while (q--) {
        scanf("%s", t);
        int len = strlen(t);
        for (int i = len; i < (len << 1) - 1; i++) t[i] = t[i - len];
        LCS(t, q + 1, len);
    }
	return 0;
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
水资源是人类社会的宝贵财富,在生活、工农业生产中是不可缺少的。随着世界人口的增长及工农业生产的发展,需水量也在日益增长,水已经变得比以往任何时候都要珍贵。但是,由于人类的生产和生活,导致水体的污染,水质恶化,使有限的水资源更加紧张。长期以来,油类物质(石油类物质和动植物油)一直是水和土壤中的重要污染源。它不仅对人的身体健康带来极大危害,而且使水质恶化,严重破坏水体生态平衡。因此各国都加强了油类物质对水体和土壤的污染的治理。对于水中油含量的检测,我国处于落后阶段,与国际先进水平存在差距,所以难以满足当今技术水平的要求。为了取得具有代表性的正确数据,使分析数据具有与现代测试技术水平相应的准确性和先进性,不断提高分析成果的可比性和应用效果,检测的方法和仪器是非常重要的。只有保证了这两方面才能保证快速和准确地测量出水中油类污染物含量,以达到保护和治理水污染的目的。开展水中油污染检测方法、技术和检测设备的研究,是提高水污染检测的一条重要措施。通过本课题的研究,探索出一套适合我国国情的水质污染现场检测技术和检测设备,具有广泛的应用前景和科学研究价值。 本课题针对我国水体的油污染,探索一套检测油污染的可行方案和方法,利用非分散红外光度法技术,开发研制具有自主知识产权的适合国情的适于野外便携式的测油仪。利用此仪器,可以检测出被测水样中亚甲基、甲基物质和动植物油脂的污染物含量,为我国众多的环境检测站点监测水体的油污染状况提供依据。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值