题意

给定两个字符串 A 和 B,求下面四个问题的答案:

1.在 A 的子串中,不是 B 的子串的字符串的数量。

2.在 A 的子串中,不是 B 的子序列的字符串的数量。

3.在 A 的子序列中,不是 B 的子串的字符串的数量。

4.在 A 的子序列中,不是 B 的子序列的字符串的数量。

其中子串是指本质不同的子串,不同的位置相同的串也只算一个串

|S|<=2000

分析

构造俩自动机然后同时跑

题解

构造一个子序列自动机,再构造一个后缀自动机,然后从根依次转移,记录状态上的信息,记忆化一下。

#include <bits/stdc++.h>
using namespace std;
const int mo=1e9+7, Lim=100005;
int nid;
struct node {
node *c[26], *f;
int len, s, id;
bool flag;
void init() {
memset(c, 0, sizeof c);
f=0;
len=s=0;
flag=0;
}
}Po[Lim], *iT=Po, *root[2], *last[2], *rt[2];
node *newnode() {
iT->init();
return iT++;
}
void init() {
root[0]=newnode();
root[1]=newnode();
last[0]=root[0];
last[1]=root[1];
rt[0]=newnode();
rt[1]=newnode();
}
void add1(int who, int ch) {
node *now=last[who], *x=newnode();
last[who]=x;
x->len=now->len+1;
x->id=nid++;
for(; now && !now->c[ch]; now=now->f) {
now->c[ch]=x;
}
if(!now) {
x->f=root[who];
return;
}
node *y=now->c[ch];
if(y->len==now->len+1) {
x->f=y;
return;
}
node *z=newnode();
*z=*y;
z->id=nid++;
z->len=now->len+1;
x->f=y->f=z;
for(; now && now->c[ch]==y; now=now->f) {
now->c[ch]=z;
}
}
void cal(node *x) {
if(x->flag) {
return;
}
x->flag=1;
x->s=1;
for(int ch=0; ch<26; ++ch) {
if(x->c[ch]) {
cal(x->c[ch]);
x->s+=x->c[ch]->s;
if(x->s>=mo) {
x->s-=mo;
}
}
}
}
void build1(int who, char *s) {
nid=1;
for(; *s; ++s) {
add1(who, *s-'a');
}
cal(root[who]);
}
void build2(int who, char *s) {
nid=1;
static node *tc[26];
memset(tc, 0, sizeof tc);
for(; *s; --s) {
node *now=newnode();
now->id=nid++;
for(int ch=0; ch<26; ++ch) {
now->c[ch]=tc[ch];
}
tc[*s-'a']=now;
}
for(int ch=0; ch<26; ++ch) {
rt[who]->c[ch]=tc[ch];
}
cal(rt[who]);
}
int vis[5005][5005];
int getans(node *x, node *y) {
if(!x) {
return 0;
}
if(!y) {
return x->s;
}
if(vis[x->id][y->id]!=-1) {
return vis[x->id][y->id];
}
int ret=0;
for(int ch=0; ch<26; ++ch) {
ret+=getans(x->c[ch], y->c[ch]);
if(ret>=mo) {
ret-=mo;
}
}
return vis[x->id][y->id]=ret;
}
char s[2][2005], *it[2];
int main() {
init();
scanf("%s%s", s[0], s[1]);
build1(0, s[0]);
build1(1, s[1]);
for(int i=0; i<2; ++i) {
for(it[i]=s[i]; *(it[i]+1); ++it[i]);
}
build2(0, it[0]);
build2(1, it[1]);
memset(vis, -1, sizeof vis); printf("%d\n", getans(root[0], root[1]));
memset(vis, -1, sizeof vis); printf("%d\n", getans(root[0], rt[1]));
memset(vis, -1, sizeof vis); printf("%d\n", getans(rt[0], root[1]));
memset(vis, -1, sizeof vis); printf("%d\n", getans(rt[0], rt[1]));
return 0;
}

弱省互测#0 t2的更多相关文章

  1. 弱省互测#2 t2

    题意 给两个树,大小分别为n和m,现在两棵树各选一些点(包括1),使得这棵树以1号点为根同构(同构就是每个点的孩子数目相同),求最大的同构树.(n, m<=500) 分析 我们从两棵树中各取出一 ...

  2. 弱省互测#0 t3

    Case 1 题意 要求给出下面代码的答案然后构造输入. 给一个图, n 个点 m 条边 q 次询问,输出所有点对之间最大权值最小的路径. 题解 把每一个询问的输出看成一条边,建一棵最小生成树. Ca ...

  3. 弱省互测#0 t1

    题意 给一个\(N \times M\)的01网格,1不能走,从起点\((1, 1)\)走到\((N, M)\),每次只能向下或向右走一格,问两条不相交的路径的方案数.(n, m<=1000) ...

  4. 【CH 弱省互测 Round #1 】OVOO(可持久化可并堆)

    Description 给定一颗 \(n\) 个点的树,带边权. 你可以选出一个包含 \(1\) 顶点的连通块,连通块的权值为连接块内这些点的边权和. 求一种选法,使得这个选法的权值是所有选法中第 \ ...

  5. 弱省互测#2 t3

    题意 给出\(n\)个01字节和\(m\)个01字节,要求用后者去匹配前者,两个串能匹配当且仅当除了每个字节末位不同,其他位都要相同.问匹配后者至少有多少个末位不同.(\(1 \le m \le n ...

  6. 弱省互测#1 t3

    题意 给出一棵n个点的树,求包含1号点的第k小的连通块权值和.(\(n<=10^5\)) 分析 k小一般考虑堆... 题解 堆中关键字为\(s(x)+min(a)\),其中\(s(x)\)表示\ ...

  7. 【2018集训队互测】【XSY3372】取石子

    题目来源:2018集训队互测 Round17 T2 题意: 题解: 显然我是不可能想出来的……但是觉得这题题解太神了就来搬(chao)一下……Orzpyz! 显然不会无解…… 为了方便计算石子个数,在 ...

  8. 【loj2461】【2018集训队互测Day 1】完美的队列

    #2461. 「2018 集训队互测 Day 1」完美的队列 传送门: https://loj.ac/problem/2461 题解: 直接做可能一次操作加入队列同时会弹出很多数字,无法维护:一个操作 ...

  9. 洛谷 P4463 - [集训队互测 2012] calc(多项式)

    题面传送门 & 加强版题面传送门 竟然能独立做出 jxd 互测的题(及其加强版),震撼震撼(((故写题解以祭之 首先由于 \(a_1,a_2,\cdots,a_n\) 互不相同,故可以考虑求出 ...

随机推荐

  1. 再说表单验证,在Web Api中使用ModelState进行接口参数验证

    写在前面 上篇文章中说到了表单验证的问题,然后尝试了一下用扩展方法实现链式编程,评论区大家讨论的非常激烈也推荐了一些很强大的验证插件.其中一位园友提到了说可以使用MVC的ModelState,因为之前 ...

  2. file_get_contents带bom

    $dmText = file_get_contents( AROOT .'data' . DS . 'DMType.json.php'); if(preg_match('/^\xEF\xBB\xBF/ ...

  3. php中计算二维数组中某一元素之和

    [0] => array(5) { ["id"] => string(2) "11" ["name"] => string ...

  4. Burp Suite新手指南

    Burp Suite想必大家都用过,但是大家未必知道它的所有功能.因此,本文的主要目的就是尽量深入介绍各种功能.BurpSuite有以下这些功能: 截获代理– 让你审查修改浏览器和目标应用间的流量. ...

  5. Java课程设计--山寨版QQ

    Java课设要求做一个聊天通讯窗口,索性直接照着QQ撸了一个Demo出来 界面高仿qq,初学JAVA,技术比较渣,不喜勿喷 线程开的比较多性能不是太好,一般电脑开两个客户端聊天就卡卡的 先上图 使用方 ...

  6. Python全栈开发【模块】

    Python全栈开发[模块] 本节内容: 模块介绍 time random os sys json & picle shelve XML hashlib ConfigParser loggin ...

  7. 关于python的10个建议,比较适合新手吧.

    关于python的十个建议 http://safehammad.com/downloads/python-idioms-2014-01-16.pdf

  8. Appium for iOS setup

    windows下appium设置 之前研究了一段时间的appium for native app 相应的总结如下:                                           ...

  9. Python 小爬虫流程总结

    接触Python3一个月了,在此分享一下知识点,也算是温故而知新了. 接触python之前是做前端的.一直希望接触面能深一点.因工作需求开始学python,几乎做的都是爬虫..第一个demo就是爬取X ...

  10. 一键启动NameNode和DataNode--shell脚本

    使用shell脚本,一键启动hadoop中的NameNode和DataNode.分为普通版和装逼版.装逼版较普通版多了很多判断和信息提示,当然主要还是为了我联系shell脚本而写的. 如果想实现复用, ...