月月查华华的手机 序列自动机

链接:https://ac.nowcoder.com/acm/contest/392/J
来源:牛客网
 

题目描述

月月和华华一起去吃饭了。期间华华有事出去了一会儿,没有带手机。月月出于人类最单纯的好奇心,打开了华华的手机。哇,她看到了一片的QQ推荐好友,似乎华华还没有浏览过。月月顿时醋意大发,出于对好朋友的关心,为了避免华华浪费太多时间和其他网友聊天,她要删掉一些推荐好友。但是为了不让华华发现,产生猜疑,破坏了他们的友情,月月决定只删华华有可能搭讪的推荐好友。
月月熟知华华搭讪的规则。华华想与某个小姐姐搭讪,当且仅当小姐姐的昵称是他的昵称的子序列。为了方便,华华和小姐姐的昵称只由小写字母构成。为了更加方便,保证小姐姐的昵称长度不会比华华的长。
现在月月要快速的判断出哪些推荐好友要删掉,因为华华快回来了,时间紧迫,月月有点手忙脚乱,所以你赶紧写个程序帮帮她吧!

输入描述:

第一行输入一个字符串A表示华华的昵称。
第二行输入一个正整数N表示华华的推荐好友的个数。
接下来N行,每行输入一个字符串Bi

题解:求一个模式串的子序列有没有出现过匹配串的这种题目可以用序列自动机来求解

自己画一个二维矩阵模拟一下代码的过程就能明白什么意思了

大概就是每一层往下都会增加一个新的出现过的字母,然后从下往上通过字母来回溯,能回溯到最顶层说明有这个子序列出现过

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<cstring>
#include<string>
#include<vector>
#include<stack>
#include<map>
#include<queue>
#include<set>
#include<ctime>
#include<cstdlib>
#include<limits>
using namespace std;
const double eps=1.0e-5;
const int maxn=1000000+10;
 
int m,n,t;
char a[maxn],b[maxn];
int nxt[maxn][30];
 
int main()
{
		scanf("%s",a+1);int n=strlen(a+1);
		for(int i=n;i>=1;i--){
			for(int j=0;j<26;j++) nxt[i-1][j]=nxt[i][j];
			nxt[i-1][a[i]-'a']=i;
		}
		scanf("%d",&t);
		while(t--){
			scanf("%s",b+1);m=strlen(b+1);
			int next=0,flag=1;
			for(int i=0;i<m;i++){
				next=nxt[next][b[i+1]-'a'];
				if(next==0){
					puts("No");
					flag=false;
					break;
				}
			}
			if(flag) puts("Yes");
		}
	
}

然后还找到了,比较规范化一点的模板

#include<bits/stdc++.h>
using namespace std;
int nxt[1000006][27],now[27],n;
string s,t;
int main(){
    fill(now,now+27,-1);
    cin>>s;
    int len=s.size();
    for(int i=len-1;i>=0;i--){
        for(int j=0;j<26;j++)
            nxt[i][j]=now[j];
        now[s[i]-'a']=i;
    }//序列自动机初始化
    cin>>n;
    while(n--){
        cin>>t;
        bool flag=true;
        len=t.size();
        int lac=now[t[0]-'a'];
        if(lac==-1){printf("No\n");}
        else {
            for(int i=1;i<len;i++){
                lac=nxt[lac][t[i]-'a'];
                if(lac==-1){
                    flag=false;
                    break;
                }
            }
            if(flag)printf("Yes\n");
            else printf("No\n");
        }
    }
    return 0;
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值