2019南昌邀请赛网络预选赛 M. Subsequence
M. Subsequence(思维)
•题意
给出一个只包含小写字母的字符串 $s$;
接下来给你和 n 个 串 $t_{1,2,...,n}$,判断第 i 个串 $t_i$ 是否为 串 s 的子序列;
如果是,输出"YES",反之,输出"NO";
•题解
将 $s$ 中 26 个字母出现的位置分别记录下来;
我放在了 vector<> 中;
- int n=strlen(s+);///个人习惯,s下标从1开始
- for(int i=;i <= n;++i)
- v[s[i]-'a'].push_back(i);
假设当前要找的串 $t$ 长度为 m;
那么,在匹配 $t_1$ 时,只需判断一下 $v[t_1 -\ 'a'].size()$ 是否大于 0 即可;
如果 > 0 ,那么 $t_1$ 就匹配 $s$ 中的 $v[t_1 -\ 'a'][0]$ 位置;
并定义 $pre=v[t_1 -\ 'a'][0]$,记录一下之前匹配好的位置;
接下来匹配 $t_2$,类似 $t_1$ 的匹配;
但是这次不仅要判断 $v[t_2 -\ 'a'].size()$ 是否大于 0,还需要找到 $v[t_2 -\ 'a']$ 是否存在 > pre 的值;
因为 $t_2$ 在 $s$ 中匹配的位置势必要在 $t_1$ 之后,如果找到,更新 pre = $t_2$ 匹配的位置;
然后,根据如上方式匹配 $t_2 , t_3 , \cdots , t_m$;
存在匹配不成功的地方,就返回 false;
我是将这个匹配方式放到了 Find() 函数里的;
定义数组 a,$a_i$ 表示 $'a'+i$ 字符在 v 中匹配到的位置;
- bool Find()
- {
- int m=strlen(t+);
- mem(a,);
- int pre=;
- for(int i=;i <= m;++i)
- {
- int x=t[i]-'a';
- ///在v[x]中查找第一个 > pre 的位置
- for(;a[x] < v[x].size() && v[x][a[x]] <= pre;a[x]++);
- ///匹配失败,返回false
- if(a[x] == v[x].size())
- return false;
- ///匹配成功,更新pre
- pre=v[x][a[x]];
- }
- return true;
- }
•Code
•坑点
起初,Find() 函数里,在 v[x] 中查找第一个 > pre 的位置时我用的方法是在 v[x] 中二分查找;
- auto it=upper_bound(v[x].begin(),v[x].end(),pre);
每次判断 it 是否为 v[x].end();
一直TLE可还行;
这种题就是用来卡二分的么,可记住了;
•相似题(分割线:2019.6.21)
类比Codeforces #565C 这道题,发现,这两道题有异曲同工之妙;
首先,再分析一下本题的做法:
在 s串中找 t串,如果找到,输出 "YES",反之,输出 "NO";
如果将题意改为查找 s串 中最多有多少个 t串 呢?
在类比一下Codeforces这道题,是不是发现两者的做法一样呢?
•Code
- #include<bits/stdc++.h>
- using namespace std;
- #define memF(a,b,n) for(int i=0;i <= n;a[i++]=b);
- const int maxn=1e5+;
- char s[maxn];
- char t[maxn];
- int a[];
- vector<int >p[];///存储字母在s中出现的位置
- char *Solve()
- {
- memF(a,,);
- int len=strlen(t+);
- int cur=;
- for(int i=;i <= len;++i)
- {
- int index=t[i]-'a';
- for(;a[index] < p[index].size() && p[index][a[index]] <= cur;a[index]++);
- if(a[index] == p[index].size())
- return "NO";
- cur=p[index][a[index]];
- }
- return "YES";
- }
- void Init()
- {
- for(int i=;i < ;++i)
- p[i].clear();
- int len=strlen(s+);
- for(int i=;i <= len;++i)
- p[s[i]-'a'].push_back(i);
- }
- int main()
- {
- scanf("%s",s+);
- Init();
- int test;
- scanf("%d",&test);
- while(test--)
- {
- scanf("%s",t+);
- puts(Solve());
- }
- return ;
- }
2019南昌邀请赛网络预选赛 M. Subsequence的更多相关文章
- 2019南昌邀请赛网络预选赛 I. Max answer(单调栈+暴力??)
传送门 题意: 给你你一序列 a,共 n 个元素,求最大的F(l,r): F(l,r) = (a[l]+a[l+1]+.....+a[r])*min(l,r); ([l,r]的区间和*区间最小值,F( ...
- 2019南昌邀请赛网络预选赛 J.Distance on the tree(树链剖分)
传送门 题意: 给出一棵树,每条边都有权值: 给出 m 次询问,每次询问有三个参数 u,v,w ,求节点 u 与节点 v 之间权值 ≤ w 的路径个数: 题解: 昨天再打比赛的时候,中途,凯少和我说, ...
- POJ-2796 & 2019南昌邀请赛网络赛 I. 区间最大min*sum
http://poj.org/problem?id=2796 https://nanti.jisuanke.com/t/38228 背景 给定一个序列,对于任意区间,min表示区间中最小的数,sum表 ...
- 2019南昌邀请赛网络赛:J distance on the tree
1000ms 262144K DSM(Data Structure Master) once learned about tree when he was preparing for NOIP(N ...
- [2019南昌邀请赛网络赛D][dp]
https://nanti.jisuanke.com/t/38223 Xiao Ming recently indulges in match stick game and he thinks he ...
- 南昌邀请赛网络赛 D.Match Stick Game(dp)
南昌邀请赛网络赛 D.Match Stick Game 题目传送门 题目就会给你一个长度为n的字符串,其中\(1<n<100\).这个字符串是一个表达式,只有加减运算符,然后输入的每一个字 ...
- 2019 ICPC南昌邀请赛网络赛比赛过程及题解
解题过程 中午吃饭比较晚,到机房lfw开始发各队的账号密码,byf开始读D题,shl电脑卡的要死,启动中...然后听到谁说A题过了好多,然后shl让blf读A题,A题blf一下就A了.然后lfw读完M ...
- icpc 南昌邀请赛网络赛 Subsequence
题目链接:https://nanti.jisuanke.com/t/38232 就是判断输入是不是子序列 没想到贡献了将近十几次罚时..........可以说是菜的真实了 用cin cout超时了 改 ...
- 2019 ICPC南昌邀请赛 网络赛 K. MORE XOR
说明 \(\oplus x\)为累异或 $ x^{\oplus(a)}$为异或幂 题意&解法 题库链接 $ f(l,r)=\oplus_{i=l}^{r} a[i]$ $ g(l,r)=\ ...
随机推荐
- Hadoop Yarn框架详细解析
在说Hadoop Yarn之前,我们先来看看Yarn是怎样出现的.在古老的Hadoop1.0中,MapReduce的JobTracker负责了太多的工作,包括资源调度,管理众多的TaskTracker ...
- Typora 快捷键
今天学习了一下这个工具.很轻便,很好用的. 无序列表:输入-之后输入空格 有序列表:输入数字+“.”之后输入空格 任务列表:-[空格]空格 文字 标题:ctrl+数字 表格:ctrl+t 生成目录:[ ...
- postgreSQL备份数据
1.pg_dump 备份单一数据库 pg_dump仅导出数据库结构: pg_dump -U TestRole1 -s -f TestDb1.sql TestDb1 2.全部备份采用pg_dumpall ...
- Python爬虫之正则表达式(3)
# re.sub # 替换字符串中每一个匹配的子串后返回替换后的字符串 import re content = 'Extra strings Hello 1234567 World_This is a ...
- 英语口语练习系列-C03-常用问句
连接到英语口语系列总分类 连接到上一章抱怨 枫桥夜泊 How are you doing?你好吗? 美国人见面时候最常用的打招呼方式: "Hey! How are you doing?&qu ...
- 微信小程序——页面跳转及传参
小程序页面跳转 微信小程序的页面跳转依然是以传统的请求转发和请求重定向为主,tabbar的存在,有TAB页面的跳转. 为了微信小程序的简介方便,规定页面路径只能是十层,应尽量避免过多的交互方式. 1. ...
- DecimalFormat格式化十进制数字
DecimalFormat 是 NumberFormat 的一个具体子类,用于格式化十进制数字.该类设计有各种功能,使其能够分析和格式化任意语言环境中的数,包括对西方语言.阿拉伯语和印度语数字的支持. ...
- virtualenvwrapper 虚拟环境的使用 和 python 安装源的更改
virtualenvwrapper 虚拟环境的使用 鉴于virtualenv不便于对虚拟环境集中管理,所以推荐直接使用virtualenvwrapper. virtualenvwrapper提供了一系 ...
- No FileSystem for scheme: hdfs问题
通过FileSystem.get(conf)初始化的时候,要通过静态加载来实现,其加载类的方法代码如下: private static FileSystem createFileSystem(URI ...
- 浏览器和服务器实现跨域(CORS)判定的原理
前端对Cross-Origin Resource Sharing 问题(CORS,中文又称'跨域')应该很熟悉了.众所周知出于安全的考虑,浏览器有个同源策略,对于不同源的站点之间的相互请求会做限制(跨 ...