选拔赛 hash 字符串匹配 哈希算法(白书p374)
Description
dr所在国度的有个奇怪的规定:他们的字母不是a~z,而是用1~1000表示。
利用这个奇怪的规定,dr想出了一个好玩的游戏:首先给出n个字符串(当然每个字符用1~1000表示),然后给出有m个节点的树,节点编号1~m,这棵树以1号节点为根,每个节点都包含一个字符。现在要求用从根节点到其他m-1个节点的链上的字符组成m-1个新字符串(字符的排列顺序为从根到终点的顺序)。
是否这m-1个新字符串中的任意一个串,都与给出的n个字符串中至少一个串匹配呢?
字符串S与字符串T匹配:设len1=∣S∣,len2=∣T∣len1=|S|,len2=|T|len1=∣S∣,len2=∣T∣,则S1=T1,S2=T2,...,Slen1=Tlen1S_{1}=T_{1},S_{2}=T_{2},...,S_{len1}=T_{len1}S1=T1,S2=T2,...,Slen1=Tlen1 且 len1≤len2len1 \leq len2len1≤len2
Input
第一行输入nnn和mmm,表示有nnn个字符串,树的节点数为m(n≤103,m≤2∗105)m(n \leq 10^3,m \leq 2 * 10^5)m(n≤103,m≤2∗105)
接下来nnn行,每行首先输入一个整数kkk,代表字符串长度,然后输入kkk个整数,代表字符串Si(∣Si∣,∑∣Si∣≤2∗105)S_{i}(|Si|,\sum |Si| \leq 2 * 10^5)Si(∣Si∣,∑∣Si∣≤2∗105)
接下来一行,输入mmm个整数CiC_{i}Ci,表示树上第iii个节点上的字符
接下来m−1m-1m−1行,每行输入2个整数u,v(1≤u,v≤m)u,v(1 \leq u,v \leq m)u,v(1≤u,v≤m),表示uuu和vvv有一条边
Output
输出占一行,都能匹配输出YES,否则输出NO
Sample Input 1
1 4
4 1 2 3 4
1 2 3 2
1 2
2 3
1 4
Sample Output 1
YES 思路:运用hash算法将字符串进行哈希,并用unordered_map储存(因为哈希的值会很大,有ull那么大,用普通的数组存不下,故考虑用map,而且没必要排序
,所以用unordered_map提高效率。ps:以前没用过map,这几次遇到的题用map都巨方便,运用桶排序的思维,且可以查询任意类型)。然后用dfs查询所有
字典树出现的串的情况就OK了。
贴一发文钧菊苣的代码
#include <bits/stdc++.h>
#define pb push_back
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
typedef vector<int> VI;
//head
const int MX=2e5+;
const int has=;
const double eps=1e-;
int n,m;
int val[MX];
VI E[MX];
unordered_map<ull,int>vis;
int dfs(int u,int fa,ull h){
int res=;//在保证了根节点满足条件的情况下的初始值
for(auto v:E[u]){
if(v==fa) continue;
if(!vis[h*has+val[v]]){
return ;
}
res&=dfs(v,u,h*has+val[v]);
}
return res;
} int main(){
scanf("%d%d",&n,&m);
for(int i=;i<n;i++){//字符串总数
int k;scanf("%d",&k);
ull h=;
for(int j=;j<k;j++){
int x;scanf("%d",&x);//每串字符的内容
h=h*has+x;
vis[h]=;//hash每串字符的前缀串并标记对应hash值
}
}
for(int i=;i<=m;i++) //对应字符
scanf("%d",&val[i]);
for(int i=;i<m;i++){//树结构
int u,v;scanf("%d%d",&u,&v);
E[u].pb(v);E[v].pb(u);//建树
}
ull h=val[];
if(!vis[h]) puts("NO");
else{
if(dfs(,-,h)) puts("YES");
else puts("NO");
}
return ;
}
选拔赛 hash 字符串匹配 哈希算法(白书p374)的更多相关文章
- Hash——字符串匹配(求s1在s2中出现的次数)
题目描述: 这是一道模板题. 给定一个字符串 A 和一个字符串 B ,求 B 在 A 中的出现次数.A 和 B中的字符均为英语大写字母. 求A 在 B 中出现了几次.(可重叠) 样例输入: 3 BA ...
- 实现字符串匹配的KMP算法
KMP算法是Knuth-Morris-Pratt算法的简称,它主要用于解决在一个长字符串S中匹配一个较短字符串s. 首先我们从整体来把我这个算法的思想. 字符串匹配的朴素算法: 我们容易想到朴素算法, ...
- Luogu 3375 【模板】KMP字符串匹配(KMP算法)
Luogu 3375 [模板]KMP字符串匹配(KMP算法) Description 如题,给出两个字符串s1和s2,其中s2为s1的子串,求出s2在s1中所有出现的位置. 为了减少骗分的情况,接下来 ...
- 字符串匹配的 Boyer-Moore 算法
上一篇文章,我介绍了 字符串匹配的KMP算法 但是,它并不是效率最高的算法,实际采用并不多.各种文本编辑器的” 查找” 功能(Ctrl+F),大多采用 Boyer-Moore 算法. 下面,我根据 M ...
- 字符串匹配的 KMP算法
一般字符串匹配过程 KMP算法是字符串匹配算法的一种改进版,一般的字符串匹配算法是:从主串(目标字符串)和模式串(待匹配字符串)的第一个字符开始比较,如果相等则继续匹配下一个字符, 如果不相等则从主串 ...
- 字符串匹配的kmp算法 及 python实现
一:背景 给定一个主串(以 S 代替)和模式串(以 P 代替),要求找出 P 在 S 中出现的位置,此即串的模式匹配问题. Knuth-Morris-Pratt 算法(简称 KMP)是解决这一问题的常 ...
- HDU 1711 Number Sequence (字符串匹配,KMP算法)
HDU 1711 Number Sequence (字符串匹配,KMP算法) Description Given two sequences of numbers : a1, a2, ...... , ...
- 字符串匹配(KMP 算法 含代码)
主要是针对字符串的匹配算法进行解说 有关字符串的基本知识 传统的串匹配法 模式匹配的一种改进算法KMP算法 网上一比較易懂的解说 小样例 1计算next 2计算nextval 代码 有关字符串的基本知 ...
- 分布式_理论_08_Consistent Hash(一致性哈希算法)
一.前言 五.参考资料 1.分布式理论(八)—— Consistent Hash(一致性哈希算法)
随机推荐
- day66-CSS伪类选择器和伪元素选择器
1. 伪类选择器:hover 和 focus 比较常用. 1.1 hover:把鼠标移动到内容迈腾2020款TSI DSG舒适型的时候,字体变成了红色. html: <body> < ...
- Anaconda 安装 TensorFlow ImportError:DLL加载失败,错误代码为-1073741795
错误再现 环境: 使用Anaconda 中 conda 4.6.2, Python 3.7版本 Windows 7 操作系统 CPU: Intel i5 原始安装过程 直接在CMD中,安装链接 中的方 ...
- 牛客寒假算法基础集训营1B题
链接:https://ac.nowcoder.com/acm/contest/317/B 来源:牛客网 题目描述 小a非常喜欢204204这个数字,因为′a′+′k′=204′a′+′k′=204. ...
- scp 碰到的问题
将 nodejs 的全局目录scp复制到另外一台机器部署代码, 发现运行报错, 提示缺少依赖模块. 检查了很久, 没发现问题. 后来发现,软链接 scp后不再是软链接而是对应文件, 导致相对路径改变!
- @ResponseBody与@RestController
@ResponseBody与@RestController的作用与区别 https://blog.csdn.net/xfl4629712/article/details/78528387
- remove_if 的效率测试
#include <iostream> #include <functional> #include <vector> #include <algorithm ...
- Java零基础学习详解
01DButils工具类的介绍个三个核心类 * A: DButils工具类的介绍个三个核心类 * a: 概述 * DBUtils是java编程中的数据库操作实用工具,小巧简单实用. * DBUtils ...
- 1051: [HAOI2006]受欢迎的牛 (tarjan强连通分量+缩点)
题目大意:CodeVs2822的简单版本 传送门 $Tarjan$强连通分量+缩点,若连通块的个数等于一则输出n:若缩点后图中出度为0的点个数为1,输出对应连通块内的点数:否则输出0: 代码中注释部分 ...
- sha1和md5的区别
Hash,一般翻译做"散列",也有直接音译为"哈希"的,就是把任意长度的输入,变换成固定长度的输出,该输出就是散列值.这种转换是一种压缩映射,也就是,散列值的空 ...
- 缩写: i = i + 1 和 i += 1,可以看做是 i 自加的值 是1。
i += 1; 这样有助于记忆: i自加的值等于1