字符串哈希算法

哈希算法

字符串的哈希算法,通俗的理解,就是将一个字符串,转化成整数
原来我们进行字符串匹配的时候,就是一个个去匹配,那么时间复杂度是o(n),如果转化成数字,去匹配那么时间复杂度会变成o(1)。

哈希算法的引入

首先联想一下二进制数,对于任意一个二进制数,我们将它化为10进制的数的方法如下(以二进制数1101101为例):
1101101=1 * 2 ^ 6+1 * 2 ^ 5+0 * 2 ^ 4+1 * 2 ^ 3+1 * 2 ^ 2+1*2 ^1+1
这是二进制转化为十进制
那么如果我们想把字符串转化成十进制的整数,那么我们是不是就可以把这个字符串设为p进制的数,然后将a~z的字母设为1 ~26,那么我们在进行转化的时候是不是就可以按照二进制转换成十进制那样那么哈希就是经过一个转换,再模上一个q,就是哈希的解法了。
在这里插入图片描述
上图是我们以字符串abc为示范
得到的结果就是我们想要的整数,但是在有时候,难免会出现,两个字符串不相同但是整数相同的情况,为了尽量减少这种情况,我们把p往往取为131或者13331,然后q我们一般取2的64次方,因为我们把哈希算出来的整数存放在unsigned long long 中,而这个类型的数据溢出就相当于系统自动帮你进行2的64次方取模运算,所以我们在算哈希值的时候,就可以不用取模了。

哈希值

我们在算一个字符串的哈希值的时候,会把前缀的哈希值都算出来
在这里插入图片描述
我们再以字符串abc为例,通过观察,我们可以看出,ab的哈希值等于a的哈希值乘上一个p再加b的值,就相当于把原本在个位的a移到十位上,把b加到个位里,所以我们可以得到一个式子:
h[i]=h[i-1]*p+s[i]-‘a’+1;
后面的这个s[i]-‘a’+1,就相当于a=1,b=2以此类推。

部分代码:

for(int i=1;i<=strlen(a);i++)
{
	h[i]=h[i-1]*p+s[i]-'a'+1}//a是字符串数组,h是存放哈希值的

求子串的哈希值

如果知道我们1到r的哈希值,让我们求h到r的哈希值
这个求的过程有点类似于给你一个数比如210,让你求个位加十位的值
在这里插入图片描述
那么我们也可以得到一个式子w[h-r]=h[h]-h[r]*q ^ (h-r+1);
为了减少时间复杂度我们一般用一个数组q直接保存q的次方值
代码如下

q[0]=1;
for(i=1;i<=strlen(a);i++)
    {
        q[i]=q[i-1]*base;
    }

一般的总代码

#include<stdio.h>
#include<string.h>
char a[1000010];
unsigned long long h[1000010],q[1000010];//一定要注意数据类型
int main()
{
    int base=131;//将p取为131
    scanf("%s",a+1);
    int n,m,i;
    m=strlen(a+1);
    h[0]=0;
    q[0]=1;
    for(i=1;i<=m;i++)//从1开始
    {
        h[i]=h[i-1]*base+(a[i]-'a'+1);
        q[i]=q[i-1]*base;
    }

以上代码就是求哈希值的代码。

  • 15
    点赞
  • 41
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
字符串哈希算法是一种将字符串映射为数字的算法,常用于字符串的比较和匹配。在C++中,可以使用字符串哈希算法来加速字符串的比较操作。 引用\[1\]中的代码示例展示了一个使用字符串哈希算法的C++代码。该代码使用了前缀和数组和字符串数组来存储字符串,并通过计算哈希值来比较两个子串是否相等。其中,哈希值的计算使用了前缀和数组和幂运算。 引用\[2\]中的解释指出,使用字符串哈希的目的是为了比较字符串时不直接比较字符串本身,而是比较它们对应映射的数字。这样可以将子串的哈希值的时间复杂度降低到O(1),从而节省时间。 引用\[3\]中的代码示例也展示了一个使用字符串哈希算法的C++代码。该代码使用了前缀和数组和字符串数组来存储字符串,并通过计算哈希值来比较两个子串是否相等。与引用\[1\]中的代码类似,哈希值的计算也使用了前缀和数组和幂运算。 综上所述,字符串哈希算法是一种将字符串映射为数字的算法,常用于字符串的比较和匹配。在C++中,可以使用前缀和数组和幂运算来计算字符串的哈希值,并通过比较哈希值来判断两个子串是否相等。 #### 引用[.reference_title] - *1* [C++算法题 # 33 字符串哈希](https://blog.csdn.net/weixin_44536804/article/details/123425533)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [字符串哈希(c++)](https://blog.csdn.net/qq_41829492/article/details/120980055)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [AcWing 841. 字符串哈希(C++算法)](https://blog.csdn.net/YSA__/article/details/108453403)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值