我是靠谱客的博主 失眠灰狼,最近开发中收集的这篇文章主要介绍子串查找【哈希算法】子串查找,觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

子串查找

传送门: 链接   来源: UPC8170

题目描述

给定一个字符串A和一个字符串B,求B在A中的出现次数。A和B中的字符均为英语大写字母或小写字母。
A中不同位置出现的B可重叠。

输入

输入共两行,分别是字符串A和字符串B。

输出

输出一个整数,表示B在A中的出现次数。

样例输入

  zyzyzyz

  zyz

样例输出

  3

提示

1≤A,B的长度≤106,A、B仅包含大小写字母。

思路:

之前总结过string的函数和它的很多用法,直接用string暴力(反正b的长度知道,只需要枚举起点即可),结果T了。

(知道string耗时高,但也太高了吧!!)

TLE的代码:

#include<bits/stdc++.h>
using namespace std;
int main()
{
    string a,b;
    cin>>a>>b;
    int la=a.size(),lb=b.size();
    int cnt=0;
    for(int i=0;i<la;i++){
        if(a[i]==b[0]&&i+lb<=la){
            string c(a.begin()+i,a.begin()+i+lb);
            //cout<<c<<endl;
            if(b==c) cnt++;
        }
    }
    cout<<cnt;
    return 0;
}

就用hash写了,hash之前没接触过,看了讲解之后感觉就是把密码加密其它字符的一个过程,它的密码本有很多种,所以如果出现加密后相同的结果就要更换加密方式了,不然会被hack掉。

我用的这个模板来自: 任小喵r

里面用的转化方法是化成k进制(k=26 好像不太容易被hack)即进制哈希,但里面没有对求出的hash值取模所以肯定会爆long long,如果单纯是用hash的话,爆就让它爆了也不会影响结果,但要做类似记录一个hash值并让它与后面的数比较那就要取模了,不然很可能wa。

AC代码:

#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
const LL MAX=1e6;
LL ha[MAX+5];
LL k=26;
LL qpow(LL m,LL q)
{
    LL ans=1;
    while(q){
        if(q%2){
            ans*=m;
        }
        m*=m;
        q/=2;
    }
    return ans;
}
LL hashb(string b)
{
    LL ans=0,lb=b.size();
    for(LL i=lb-1;i>=0;i--)
        ans=ans*k+b[i];
    return ans;
}

void hasha(string a)
{
    LL la=a.size();
    for(LL i=la-1;i>=0;i--){
        ha[i]=ha[i+1]*k+a[i];
    }
}

int main()
{
    string a,b;
    cin>>a>>b;
    LL la=a.size(),lb=b.size();
    LL hb=hashb(b);
    hasha(a);
    LL num=qpow(k,lb),ans=0;
    for(LL i=0;i<=la-lb;i++){
        if(hb==ha[i]-ha[i+lb]*num){ ///要是不理解就把k换成10,自己出样例模拟一下就懂了
            ans++;
        }
    }
    cout<<ans<<endl;
    return 0;
}

 

最后

以上就是失眠灰狼为你收集整理的子串查找【哈希算法】子串查找的全部内容,希望文章能够帮你解决子串查找【哈希算法】子串查找所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(39)

评论列表共有 0 条评论

立即
投稿
返回
顶部