[BZOJ3507]通配符匹配
3507: [Cqoi2014]通配符匹配
Time Limit: 10 Sec Memory Limit: 128 MB
Description
几乎所有操作系统的命令行界面(CLI)中都支持文件名的通配符匹配以方便用户。最常见的通配符有两个,一个
是星号(“”’),可以匹配0个及以上的任意字符:另一个是问号(“?”),可以匹配恰好一个任意字符。
现在需要你编写一个程序,对于给定的文件名列表和一个包含通配符的字符串,判断哪些文件可以被匹配。
Input
第一行是一个由小写字母和上述通配符组成的字符串。
第二行包含一个整数n,表示文件个数。
接下来n行,每行为一个仅包含小写字母字符串,表示文件名列表。
Output
输出n行,每行为“YES”或“NO”,表示对应文件能否被通配符匹配。
Sample Input
6
acaacatctc
acatctc
aacacatctc
aggggcaacacctc
aggggcaacatctc
aggggcaacctct
Sample Output
YES
YES
YES
YES
NO
HINT
对于1 00%的数据
·字符串长度不超过1 00000
· 1 <=n<=100
·通配符个数不超过10
题解:
考场打暴力打炸了……
不过在后来改题的时候我发现了暴力思路的一个大问题……我们还是直接讲正解吧
题目给的通配符有2种,一种是’*’,它可以伸展;一种是‘?’,它的长度固定;
不难发现,给我们的处理带来难度的是'*',因为他的长度不确定;
但转念一想,如果除了*,其他的都能匹配,那*也就能匹配了.
注意到通配符个数<=10,那么我们可以先按照*,把原串分成一些包含'?'和字母的段.
而对于每一段,只要其中的所有小字符串成功匹配,并且符合'?'对长度的要求,这一段就匹配成功了
所以我们再在每一段里,按照'?'再分为几小块只含字母的串,并且分别求出hash值;同时求出文本串的hash值.
在匹配的时候,对于某一段,枚举起点,看在这一段在哪里可以匹配.
由于对于段间的枚举,影响因素只有*,所以匹配点肯定越靠前越好,因此这里可以贪心,即找到最靠前的匹配点即可停止.
而对于段内比较,利用刚才求的hash值比较就好.
代码见下:
#include<cstdio>
#include<cstring>
#include<iostream>
#include<string>
using namespace std;
typedef unsigned long long LL;
const int N=;
const int mod1=;
int n,lens,len,cnt,cnt1[];
char s[N],text[N];
LL block[][],h[N],mi[N];
int l[][],num[];
inline LL g_hash(int l,int r){return h[r]-h[l-]*mi[r-l+];}
inline bool match(int i,int k)
{
for(int j=;j<=num[k];j++)
{
if(l[k][j]==){i++;continue;}
if(block[k][j]!=g_hash(i,i+l[k][j]-))return ;
i+=l[k][j]+;
}
return ;
}
bool judge()
{
if(!cnt)
{
if(len!=lens)return ;
for(int i=;i<=len;i++)
if(s[i]!=text[i]&&s[i]!='?')return ;
return ;
}
if(len<(cnt1[]-)+(lens-cnt1[cnt]))return ;
for(int i=;i<cnt1[];i++)if(s[i]!=text[i]&&s[i]!='?')return ;
for(int i=lens,j=len;i>cnt1[cnt];i--,j--)if(s[i]!=text[j]&&s[i]!='?')return ;
for(int i=;i<=len;i++)
h[i]=h[i-]*mod1+text[i];
int i=cnt1[],line=len-(lens-cnt1[cnt])+;
for(int k=;k<=cnt;k++)
{
for(;i<=line;i++)
if(match(i,k))break;
i+=cnt1[k]-cnt1[k-]-;
if(i>line)return ;
}
return ;
}
int main()
{
scanf("%s",s+);lens=strlen(s+);
mi[]=;for(int i=;i<=lens;i++)mi[i]=mi[i-]*mod1;
for(int i=;i<=lens;i++)
if(s[i]=='*')cnt1[++cnt]=i;
for(int i=;i<=cnt;i++)
{
num[i]=;
for(int j=cnt1[i-]+;j<cnt1[i];j++)
{
if(s[j]=='?'){num[i]++;continue;}
l[i][num[i]]++;
block[i][num[i]]=block[i][num[i]]*mod1+s[j];
}
}
scanf("%d",&n);
for(int i=;i<=n;i++)
{
scanf("%s",text+);len=strlen(text+);
if(judge())printf("YES\n");
else printf("NO\n");
}
}
BZOJ3507
[BZOJ3507]通配符匹配的更多相关文章
- [CQOI2014][bzoj3507] 通配符匹配 [字符串hash+dp]
题面 传送门 思路 0x01 KMP 一个非常显然而优秀的想法:把模板串按照'*'分段,然后对于每一段求$next$,'?'就当成可以对于任意字符匹配就行了 对于每个文本串,从前往后找第一个可以匹配的 ...
- 【BZOJ3507】通配符匹配(哈希,动态规划)
[BZOJ3507]通配符匹配(哈希,动态规划) 题面 BZOJ 题解 对于匹配唯一存在影响的只有通配符,而\(?\)的影响也并不大,所以唯一需要仔细考虑的是\(*\). 考虑一个\(dp\),设\( ...
- 【BZOJ-3507】通配符匹配 DP + Hash
3507: [Cqoi2014]通配符匹配 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 372 Solved: 156[Submit][Statu ...
- BZOJ3507 [Cqoi2014]通配符匹配
题意 几乎所有操作系统的命令行界面(CLI)中都支持文件名的通配符匹配以方便用户.最常见的通配符有两个,一个是星号("*"),可以匹配0个及以上的任意字符:另一个是问号(" ...
- BZOJ3507 [Cqoi2014]通配符匹配 【哈希 + 贪心】
题目 几乎所有操作系统的命令行界面(CLI)中都支持文件名的通配符匹配以方便用户.最常见的通配符有两个,一个 是星号(""'),可以匹配0个及以上的任意字符:另一个是问号(&quo ...
- bzoj 3507: [Cqoi2014]通配符匹配
Description 几乎所有操作系统的命令行界面(CLI)中都支持文件名的通配符匹配以方便用户.最常见的通配符有两个,一个是星号(“”’),可以匹配0个及以上的任意字符:另一个是问号(“?”),可 ...
- [Swift]LeetCode44. 通配符匹配 | Wildcard Matching
Given an input string (s) and a pattern (p), implement wildcard pattern matching with support for '? ...
- LeetCode(44): 通配符匹配
Hard! 题目描述: 给定一个字符串 (s) 和一个字符模式 (p) ,实现一个支持 '?' 和 '*' 的通配符匹配. '?' 可以匹配任何单个字符. '*' 可以匹配任意字符串(包括空字符串). ...
- [LeetCode][Facebook面试题] 通配符匹配和正则表达式匹配,题 Wildcard Matching
开篇 通常的匹配分为两类,一种是正则表达式匹配,pattern包含一些关键字,比如'*'的用法是紧跟在pattern的某个字符后,表示这个字符可以出现任意多次(包括0次). 另一种是通配符匹配,我们在 ...
随机推荐
- SRM 698 div1 RepeatString
250pts RepeatString 题意:问最少修改多少次将一个字符串修改为AA的形式.可以插入一个字符,删除一个字符,修改字符. 思路:枚举分界点,然后dp一下. /* * @Author: m ...
- Vue.js之常用指令
vue常用指令 vue.js官方给自己的定义是数据模板引擎,并给出了一套渲染数据的指令.本文详细介绍vue.js的常用指令. 官网:点我 一.v-text.v-html v-text:用于绑定文本 v ...
- Web自动化常用方法封装(不定时更新)
/** * JScript实现鼠标悬停 */public void mouseHoverJScript(By by,WebElement HoverElement) { // TODO Auto-ge ...
- [转]Git 撤销操作
二. Git撤消操作 12.1 修改最后一次提交 git commit --amend 1.新建一个文件 2.提交一个之前的更改 3.跟踪这个文件 4.跟前一次一起提交 提示你是否重新编辑提交说明,如 ...
- Elastic-Job 分布式调度平台
概述 referred:http://elasticjob.io/docs/elastic-job-lite/00-overview Elastic-Job是一个分布式调度解决方案,由两个相互独立的子 ...
- No.03---Vue学习之路之模块化组织
前两篇讲解了一下 Vuex 的基本使用方法,可是在实际项目中那么写肯定是不合理的,如果组件太多,不可能把所有组件的数据都放到一个 store.js 中的,所以就需要模块化的组织 Vuex,首先看一下 ...
- 【jpeg_Class 类】使用说明
jpeg_Class类是针对图片操作类,可以获取图片属性.等比例缩略图片.裁切图片.图片上打印文字及打印水印等功能. 目录 原型 参数 返回 说明 Sub load(byVal path) path ...
- 设计模式C++实现
准备写一系列笔记用来记录学习设计模式的过程,同时写出自己对几种主要的设计模式的理解,以及编码实现,同时总结. 主要参考书籍就是 <Head First Design Patterns>这本 ...
- XSS 注入检查点
如果你有个论坛,一般你会很注意用户发帖的注入问题,往往这个地方不会被注入,因为开发特别照顾.原则上XSS都是用户输入的,但是许多边角还是容易忽略.枚举一些检查点. 分页 分页通用组件获取url,修改p ...
- 算法笔记(c++)--01背包问题
算法笔记(c++)--经典01背包问题 算法解释起来太抽象了.也不是很好理解,最好的办法就是一步步写出来. 背包问题的核心在于m[i][j]=max(m[i-1][j],m[i-1][j-w[i]]+ ...