P3167 [CQOI2014]通配符匹配 题解
题目大意
给出一个字符串,其中包含两种通配符 ‘?’和 ‘*’ ,‘?’可以代替一个字符,‘*’可以代替一个字符串(长度可以为0)
然后给出几个字符转,判断能否用给出的字符串表示出来
样例解释
给出字符串:*aca?ctc
判断字符串 :acaacatctc
此时用 * 代替 aba 用 ?代替了t,所以输出YES
解题思路
dp + hash
用hash预处理出字符串的前缀和, 方便判断两端字符是否相同
假设 给出字符串 为str1, 要判断的字符串为 str2
f[i][j]表示 第i个通配符与 要str2的前i个字符 匹配是否可行
当第i个通配符是 * 时特殊判断:
因为 * 可以匹配任意长度的字符串,所以 f[i][j] |=f[i][j-1] (如果匹配到 j-1,那让*再匹配一个,一定可行)
然后枚举str2的每个字符,下标为j,
如果f[i][j]可行,继续判断 给出的字符串 中不是通配符(已经固定)的部分是否能匹配,进行转移
注意
1、因为结尾没有通配符,所以在 str1 结尾加上一个 ‘?’, 在每个str2结尾加上任意一个字符,便于处理结尾部分、
2、如果第i个通配符是 * 应将f[i+1][rb] 赋值为 true,(应为*可以是长度为0的,具体看代码注释)
惊喜
这题与洛谷 P2536 [AHOI2005]病毒检测 基本一样(要统计个数,改了一下数据范围) 一份代码,两道紫题~~~
代码&注释
#include <bits/stdc++.h>
using namespace std;
#define ul unsigned long long
const ul p = 666, N = 100000;
ul sx[N+5], sy[N+5], s[N+5];
int len, lenn, w[12], num, t;
char ch[N+5], cc[N+5];
bool f[12][N+5]; // f[i][j] 表示第i个通配符匹配以j结尾的字符(串)
int main(){
s[0] = 1; for (int i = 1; i <= N; i++) s[i] = s[i-1] * p;
scanf ("%s", ch + 1); len = strlen (ch + 1); ch[++len] = '?'; //在末尾加上一个 ?处理结尾部分
for (int i = 1; i <= len; i++) {
sx[i] = sx[i-1] * p + ch[i]; //预处理出 hash 前缀和
if (ch[i] == '*' or ch[i] == '?') w[++num] = i; //记录通配符的下标
}
scanf ("%d", &t);
while (t--) {
scanf ("%s", cc + 1); lenn= strlen (cc + 1);
cc[++lenn] = '!'; //在末尾加上一个 字符处理结尾部分
for (int i = 1; i <= lenn; i++) sy[i] = sy[i-1] * p + cc[i]; //预处理出 hash 前缀和
memset (f, 0, sizeof (f));
f[0][0] = 1; //初始化
for (int i = 0; i <= num; i++) {
if (ch[w[i]] == '*') for (int j = 1; j <= lenn; j++) f[i][j] |= f[i][j-1]; //是 * 特殊处理
for (int j = 0; j <= lenn; j++) {
if (!f[i][j]) continue;
int la = w[i] + 1, ra = w[i+1] - 1, lb = j + 1, rb = j + w[i+1] - w[i] - 1; //计算出两段的左右端点
if (sx[ra] - sx[la-1] * s[ra-la+1] == sy[rb] - sy[lb-1] * s[rb-lb+1]){ //判断两段是否相同
if (ch[w[i+1]] == '?') f[i+1][rb+1] = 1;
else f[i+1][rb] = 1; //注意:如果是 * ,应将f[i+1][rb]赋值为1而不是f[i+1][rb+1] (*可以匹配长度为0的字符串)
}
}
}
if (f[num][lenn]) puts ("YES");
else puts ("NO");
}
return 0;
}
P3167 [CQOI2014]通配符匹配 题解的更多相关文章
- 「题解报告」 P3167 [CQOI2014]通配符匹配
「题解报告」 P3167 [CQOI2014]通配符匹配 思路 *和?显然无法直接匹配,但是可以发现「通配符个数不超过 \(10\) 」,那么我们可以考虑分段匹配. 我们首先把原字符串分成多个以一个通 ...
- P3167 [CQOI2014]通配符匹配
吐槽 本来是去写AC自动机的,然后发现数据范围每个串100000,有100个串(???),连塞进trie树里都塞不进去,玩个鬼的AC自动机啊,tag不要乱打啊 最后拿字符串hash+爆搜一发搜过去了. ...
- BZOJ3507 [Cqoi2014]通配符匹配
题意 几乎所有操作系统的命令行界面(CLI)中都支持文件名的通配符匹配以方便用户.最常见的通配符有两个,一个是星号("*"),可以匹配0个及以上的任意字符:另一个是问号(" ...
- BZOJ3507 [Cqoi2014]通配符匹配 【哈希 + 贪心】
题目 几乎所有操作系统的命令行界面(CLI)中都支持文件名的通配符匹配以方便用户.最常见的通配符有两个,一个 是星号(""'),可以匹配0个及以上的任意字符:另一个是问号(&quo ...
- [CQOI2014]通配符匹配
Description 几乎所有操作系统的命令行界面(CLI)中都支持文件名的通配符匹配以方便用户.最常见的通配符有两个,一个是星号(""'),可以匹配0个及以上的任意字符:另一个 ...
- bzoj 3507: [Cqoi2014]通配符匹配
Description 几乎所有操作系统的命令行界面(CLI)中都支持文件名的通配符匹配以方便用户.最常见的通配符有两个,一个是星号(“”’),可以匹配0个及以上的任意字符:另一个是问号(“?”),可 ...
- 【bzoj3570】 Cqoi2014—通配符匹配
http://www.lydsy.com/JudgeOnline/problem.php?id=3507 (题目链接) 题意 给出一个主串,里面有些通配符,'*'可以代替任意字符串或者消失,'?'可以 ...
- [BZOJ3507][CQOI2014]通配符匹配(DP+Hash)
显然f[i][j]表示S匹配到第i个通配符,T匹配到第j个字符,是否可行. 一次一起转移两个通配符之间的所有字符,Hash判断. 稍微有点细节.常数极大卡时过排名倒数,可能是没自然溢出的原因. #in ...
- [bzoj3507 Cqoi2014]通配符匹配 (hash+DP)
传送门 Solution 显然用哈希233 设\(f[i][j]\)表示第i个通配符和当前第j个字符是否匹配 考虑两种通配符的特性,直接转移即可 Code #include <cstdio> ...
随机推荐
- 【Android 多媒体应用】使用 TTS
import java.util.Locale; import android.app.Activity; import android.os.Bundle; import android.speec ...
- 浏览器默认标签样式总结及css初始化程序(转)
浏览器默认标签样式总结及css初始化程序 html中的大部分的标签都有一些糟糕的样式,有的是标签天然自带的,有的是浏览器默认设置的,我们在写网页时,这些默认的样式就会时不时的跳出来捣一下乱,搞得我 ...
- PythonNote02_HTML标签
<!DOCTYPE> <html> <head> <meta charset = "utf-8" /> <meta name= ...
- 已看1.熟练的使用Java语言进行面向对象程序设计,有良好的编程习惯,熟悉常用的Java API,包括集合框架、多线程(并发编程)、I/O(NIO)、Socket、JDBC、XML、反射等。[泛型]\
1.熟练的使用Java语言进行面向对象程序设计,有良好的编程习惯,熟悉常用的Java API,包括集合框架.多线程(并发编程).I/O(NIO).Socket.JDBC.XML.反射等.[泛型]\1* ...
- 杭电acm 1039题
这道题也比较简单,写三个函数判断三个条件即可..... 但是开始时我按照已经注释掉的提交,居然提示WA,我百思不得其解,后改成上面的判断式就可以了,求高手解答.... #include "i ...
- Luogu 3267 [JLOI2016/SHOI2016]侦察守卫
以后要记得复习鸭 BZOJ 4557 大佬的博客 状态十分好想,设$f_{x, i}$表示以覆盖完$x$为根的子树后还能向上覆盖$i$层的最小代价,$g_{x, i}$表示以$x$为根的子树下深度为$ ...
- CodeForces 402D Upgrading Array (数学+DP)
题意:给出一个数列,可以进行一种操作将某一个前缀除去他们的gcd,有一个函数f(x),f(1) = 0 , f(x) = f(x/p)+1,f(x) = f(x/p)-1(p是坏素数), 求 sum( ...
- js/jq基础(日常整理记录)-1-纯js格式化时间
一.纯js格式化时间 之前记录了一些,工作中发现的比较常用的使用,就记录一下. 由于很基础,就直接贴出来了,不做分析了. 改造一下Date的原型 Date.prototype.format = fun ...
- [学习笔记]父进程wait和waitpid
1.wait和waitpid出现的原因 SIGCHLD q 当子进程退出的时候,内核会向父进程发送SIGCHLD信号,子进程的退出是个异步事件(子进程可以在父进程运行的任何时刻终止) q 子进程退 ...
- python 根据字典中的key,value进行排序
#coding=utf-8 import requests,json,collections,base64,datetime def sort(datas): data=json.dumps(data ...