哈希Hash在字符串中的应用_C++
本文含有原创题,涉及版权利益问题,严禁转载,违者追究法律责任
哈希大家都会用撒,字符串显然都会写撒,那么哈希离散化字符串不就懂了?!(XXX的神逻辑,其实原文是:树都晓得吧,数组显然都会开呀,那么恭喜你学会了树状数组!)
例如我们给出 n 个长度为 m 的字符串,然后给你一个长度为 m 的字符串 s ,求 s 是否在之前的 n 个串中出现过
暴力扫 n 个串,然后一位位去对,看是否相等,时间复杂度O(nm),它非常得辣鸡
当然我们可以建一颗 trie 树做,但是建树的复杂度也是O(nm)的,对于只询问一次还不如打暴力
一阵思考后,我们选用非常优秀的哈希(以下简称Hash)
(机制的人请忽略这篇文章)
平常我们用 Hash 是对数字进行处理,一般把它取模之后离散化到各个数组
但是对于字符串呢?难道把它所有位加起来取模?显然是不行的,如 ab 和 ba 这两种情况是相同的,那么小节点挂的链会很多
于是乎,我们想,对于字符串只有256种,而题目中给的一般只会包含字母,甚至是只有小写字母,瞬间压缩到26种
那么我们可以把它前面 k 位取出来,如取4位出来是 axoc,把它装换成对应的数字是 0 23 14 2 (a 看做 0,b 看做 1 …… z 看做 25)
这不就是一个二十六进制数!
把前 k 位取出来,第 i 位即为这个26进制数的第 i 位,再把它转换为10进制,如 axoc 为 45214,这样我们就得到了 Hash 的 Hash 函数了
字符串判重就容易了, Hash 具体实现过程就不用讲了吧
对于 26 进制的数,int 范围下只能取到第 6 位,long long范围下可以取到第 13 位

下面给道例题
题意很简单,给你一个串,求他有多少个不同的子串,满足前缀为A,后缀为B。 需要注意的是,串中所有的字母都是小写字母。
三个串长度均小于等于 2000,时间1s,空间128MB
样例输入:abababab
a
b
样例输出:4
直接暴力枚举然后 Hash 判重,我们这里截取前3位做 Hash 函数
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<iostream>
#include<algorithm>
using namespace std; const int N=,loc=,power[]={,,,,,,,},mo=;
struct Hash
{
int next;
char s[N];
}
hs[N*];
char a[N],b[N],s[N];
int ls,ans,next[N],f[N],top,first[mo],tg,g[mo];
bool fa[N],fb[N];
void init(char c[N],int len)
{
int i,j=;
for (i=;i<len;i++)
{
while (j&&c[i]!=c[j]) j=next[j];
if (c[i]==c[j]) j++;
next[i+]=j;
}
}
void kmp(char c[N],int len)
{
int i,j=;
top=;
for (i=;i<=len;i++) next[i]=;
init(c,len);
for (i=;i<ls;i++)
{
while (j&&s[i]!=c[j]) j=next[j];
if (s[i]==c[j]) j++;
if (j==len)
{
f[++top]=i-j+;
j=next[j];
}
}
}
void hash(char c[N],int len)
{
int i,x=,t=min(len,loc);
for (i=;i<=t;i++) x+=(c[i]-'a')*power[i];
if (first[x]) for (i=first[x];i;i=hs[i].next)
{
t=len-loc;
if (t<=) return;
for (x=;x<=t;x++) if (hs[i].s[x]!=c[loc+x]) break;
if (x>t) return;
}
if (!(hs[++top].next=first[x])) g[++tg]=x;
first[x]=top;
x=;
for (i=loc+;i<=len;i++) hs[top].s[++x]=c[i];
ans++;
}
int main()
{
int la,lb,i,j,t,x,k,l;
char c[N];
scanf("%s\n%s\n%s\n",&s,&a,&b);
ls=strlen(s);
la=strlen(a);
lb=strlen(b);
kmp(a,la);
for (i=;i<=top;i++) fa[f[i]]=;
kmp(b,lb);
for (i=;i<=top;i++) fb[f[i]]=;
t=ls-la-lb;
for (j=;j<=t;j++)
{
for (i=;i<=tg;i++) first[g[i]]=;
top=tg=;
for (i=;i<=t-j;i++)
{
x=i+j+la;
if (fa[i]&&fb[x])
{
l=;
for (k=i+la;k<x;k++) c[++l]=s[k];
hash(c,j);
}
}
}
for (i=max(la-lb-,);i<=la;i++)
{
j=i;
while (a[j]==b[j-i]) j++;
if (a[j]!='\0') continue;
for (j=;j<i;j++) c[j]=a[j];
for (k=;k<lb;k++,j++) c[j]=b[k];
kmp(c,j);
if (top) ans++;
}
printf("%d\n",ans);
return ;
}
据说可以用KMP或者EXKMP做orz
还有我的程序在 Hash 取别的位数的时候会 WA 几个点(而且还是不同的点??)
求助大神能解释一下orz(再次体现我的蒟蒻)
注:该题为原创题,可购买数据,价格 RMB 2.0
如需购买在这 Get 联系方式 http://www.cnblogs.com/hadilo/p/5932395.html
哈希Hash在字符串中的应用_C++的更多相关文章
- Redis中的哈希(Hash)
Redis 哈希(Hash) Redis hash 是一个string类型的field和value的映射表,hash特别适合用于存储对象. Redis 中每个 hash 可以存储 232 - 1 键值 ...
- Redis 命令,键(key),字符串(String),哈希(Hash),列表(List),集合(Set)(二)
Redis 命令 Redis 命令用于在 redis 服务上执行操作. 要在 redis 服务上执行命令需要一个 redis 客户端.Redis 客户端在我们之前下载的的 redis 的安装包中. ...
- Crazy Search POJ - 1200 (字符串哈希hash)
Many people like to solve hard puzzles some of which may lead them to madness. One such puzzle could ...
- 牛客练习赛33 E tokitsukaze and Similar String (字符串哈希hash)
链接:https://ac.nowcoder.com/acm/contest/308/E 来源:牛客网 tokitsukaze and Similar String 时间限制:C/C++ 2秒,其他语 ...
- hash数组快速查找一个字符串中出现最多的字符,并统计出现的次数
如何快速查找一个字符串中出现最多的字符,并统计出现的次数? 可以使用hash数组,也就是关联数组实现快速查找功能. function seek(str) { var hash = []; var ma ...
- 大话Java中的哈希(hash)结构(一)
o( ̄▽ ̄)d 小伙伴们在上网或者搞程序设计的时候,总是会听到关于“哈希(hash)”的一些东西.比如哈希算法.哈希表等等的名词,那么什么是hash呢? 一.相关概念 1.hash算法:一类特殊的算法 ...
- 面试35-删除字符串重复字符-删除出现在第二个字符串中的字符-第一个只出现一次的字符-hash表计数
#include<iostream>#include<algorithm>#include<functional>using namespace std;char ...
- 萌新笔记——Cardinality Estimation算法学习(一)(了解基数计算的基本概念及回顾求字符串中不重复元素的个数的问题)
最近在菜鸟教程上自学redis.看到Redis HyperLogLog的时候,对"基数"以及其它一些没接触过(或者是忘了)的东西产生了好奇. 于是就去搜了"HyperLo ...
- redis 哈希(hash)函数
哈希(hash)函数 hSet 命令/方法/函数 Adds a value to the hash stored at key. If this value is already in the has ...
随机推荐
- 常用js方法合集
var Default = { init: function () { }, addCookie: function (name,data) { var expdate = new Date(); / ...
- 29、phonegap入门
0. PhoneGap介绍 0.1 什么是PhoneGap? PhoneGap是一个基于HTML.CSS.JS创建跨平台移动应程序的快速开发平台.与传统Web应用不同的是,它使开发者能够利用iPho ...
- Anytime项目开发记录4
做事情列表,我在程序中命名为“正在做”. 这是一个Fragment,应用的主页面,由一个MainActivity加上DoingListFragment和PersonFragment组成.PersonF ...
- 『AngularJS』创建 Service
创建服务 Angular提供了几种有用的服务,对于所有的应用来说,你将会发现这些服务对于创建你自己的服务是有用处的.为了创建自己的服务,你应该从通过一个模块(module)注册一个服务工厂方法开始(可 ...
- 原生js实现五子棋
为什突然做这个,因为这是个笔试题,拖了一个月才写(最近终于闲了O(∩_∩)O),废话不多说,说说这个题吧 题目要求 编写一个单机[五子棋]游戏,要求如下: 1.使用原生技术实现,兼容 Chrome 浏 ...
- [Linux] umount目录提示device is busy的解决方法
使用sshfs等方式挂载的目录出现问题时,使用umount卸载经常提示device is busy,如果仔细阅读错误提示就可以找到命令lsof和fuser命令. 其实原因就是有进程占用当前目录,导致不 ...
- QQ互联登陆的最简洁代码
<?php/** * http://wiki.open.qq.com/wiki/ * Date: 14-6-18 * Time: 下午18:04 */class Model_Login_QqCo ...
- java多线程二之线程同步的三种方法
java多线程的难点是在:处理多个线程同步与并发运行时线程间的通信问题.java在处理线程同步时,常用方法有: 1.synchronized关键字. 2.Lock显示加锁. 3.信号量Se ...
- 给曾经是phper的程序员推荐个学习网站
如果你原来是一个php程序员,你对于php函数非常了解(PS:站长原来就是一个php程序员),但是现在由于工作或者其他原因要学习python,但是python很多函数我们并不清楚,在这里我给大家推荐一 ...
- bbbbbeta
about beta 写在开头:(小声bb,无任何专业知识) 好了正文开始了 = = beta冲刺对于来说可能是让我觉得非常有成就感的叭,相比于alpha,每天都能写代码的感觉真好鸭(认真脸)(虽然天 ...