【字符串算法1】 字符串Hash(优雅的暴力)

【字符串算法2】Manacher算法

【字符串算法3】KMP算法

这里将讲述  【字符串算法1】 字符串Hash

老版原文: RK哈希(Rabin_Karp 哈希) 仅仅具有参考价值

新版题目:【字符串算法1】 再谈字符串Hash(优雅的暴力)

概念

- 什么是字符串Hash?

- 就是把字符串映射成一个数字使每个字符串的映射结果不一样(把字符串有效的转化为数字

对字符进行映射

对一个字符进行唯一编码,如A-->1,B-->2,C-->3等等

一般不用ASCII码表来映射而是转化为相对小一点的值来映射

提示信息: (字符串内包含数字、大小写字母,大小写敏感)

写出val函数映射字符:

  1. int val(char ch)
  2. {
  3. if (isdigit(ch)) return(ch-''+);
  4. if (isupper(ch)) return(ch-'A'+);
  5. if (islower(ch)) return(ch-'a'+);
  6. }

对字符串进行映射

一般映射方法:hash[i]=(hash[i-1]*p+idx(s[i]))%mod (保险度:****)

hash[i]表示字符串的第i个前缀的hash值

Hash值的性质

①这样子,我们就可以记录下每个字符串对应的整数,当下一次出现了一个已经出现的字符串时,查询整数是否出现过,就可以知道 字符串是否重复出现。
②判断两个字符串是否一致,怎么办呢?直接用它们的hash值判断即可,若hash值一致,则认为字符串一致
若hash值不一致,则认为是不同的字符串。

例子

假设我们取p=13 ,mod=101
先把abc映射为一个整数
hash[0]=1,表示 a 映射为1
hash[1]=(hash[0]*p+idx(b))%mod=15,表示 ab 映射为 15
hash[2]=(hash[1]*p+idx(c))%mod=97
这样,我们就把 abc 映射为 97 这个数字了。

冲突:

假设mo数和基底e取值不当时就会发生冲突

就是两个字符串明明不同但映射出来的结果相同

举个极端的例子

e=0的情况 字符串"a"和字符串"b"映射出来的值都是0,就产生冲突

那么怎么调整才能使冲突概率小之又小呢?
- p取一个较大素数,mo取一个大素数。
习惯上,p取一个6到8位的素数即可,mo一般取大素数 1e9+7(1000000007)或
1e9+9(1000000009)【逃 19260817】

求出每个子串的hash值

注意到每一个hash[i]都是前缀和数字那么我们借用前缀和的思想,已知hash[r]和hash[l]求出Hash(l,r)表示前[l,r]子串的hash值

Hash[l]=(x1*el-1+x2*el-2+......+xl*e0)mod mo

Hash[l-1]=(x1*el-2+x2*el-3+......+xl-1*e0)mod mo

Hash[r]=(x1*er-1+x2*er-2+......+xr*e0)mod mo

Hash(l,r)=(xl*er-l+xl+1*er-l-1+......xr-1*e1+xr*e0)mod mo

Hash[l-1]*er-l+1=(x1*el-2+x2*el-3+......+xl-1*e0)*er-l+1mod mo=(x1*er-1+x2*er-2+......+xl-1*er-l+1)mod mo

Hash[r]-Hash[l-1]*er-l+1=((x1*er-1+x2*er-2+...xl-1*er-l+1+xl*er-l...+xr*e0)-(x1*er-1+x2*er-2+......+xl*er-l+1))mod mo =(xl*er-l+xl+1*er-l-1+......xr-1*e1+xr*e0)mod mo=Hash(l,r)

所以:Hash[r]-Hash[l-1]*er-l+1=Hash(l,r)

  1. ll pow(int x,int n,int p)
  2. {
  3. if (n==) return ;
  4. if (n==) return x%p;
  5. ll t=t*t%p;
  6. if (n%==) t=t*x%p;
  7. return t;
  8. }
  9. ll Hash(int l,int r)
  10. {
  11. return ((hash[r]-hash[l-]*pow(e,r-l+,mo)%mo)%mo+mo)%mo;
  12. }

其他hash方法

1. unsigned long long hash[N];
hash[i]=hash[i-1]*p(自动取模) (保险度***) 常数(几乎没有) (容易被卡)
2. hash[i]=(hash[i-1]*p+idx(s[i]))%mod (保险度****) 常数(有一点) (一般)
3. 双hash 
hash1[i]=(hash1[i-1]*p+idx(s[i]))%mod1
hash2[i]=(hash2[i-1]*p+idx(s[i]))%mod2
pair<hash1,hash2>表示一个字符串! (保险度*****) 常数(比较大)(孪生质数不可能被卡)

推荐 :hash[i]=(hash[i-1]*p+idx(s[i]))%mod (保险度****) 常数(有一点) (一般)

题目描述

如题,给定N个字符串(第i个字符串长度为Mi,字符串内包含数字、大小写字母,大小写敏感),请求出N个字符串中共有多少个不同的字符串。

输入输出格式

输入格式:

第一行包含一个整数N,为字符串的个数。

接下来N行每行包含一个字符串,为所提供的字符串。

输出格式:

输出包含一行,包含一个整数,为不同的字符串个数。

输入输出样例

输入样例#1: 复制

  1. 5
  2. abc
  3. aaaa
  4. abc
  5. abcc
  6. 12345
输出样例#1: 复制

  1. 4

说明

时空限制:1000ms,128M

数据规模:

对于30%的数据:N<=10,Mi≈6,Mmax<=15;

对于70%的数据:N<=1000,Mi≈100,Mmax<=150

对于100%的数据:N<=10000,Mi≈1000,Mmax<=1500

样例说明:

样例中第一个字符串(abc)和第三个字符串(abc)是一样的,所以所提供字符串的集合为{aaaa,abc,abcc,12345},故共计4个不同的字符串。

Tip: 感兴趣的话,你们可以先看一看以下三题:

BZOJ3097:http://www.lydsy.com/JudgeOnline/problem.php?id=3097

BZOJ3098:http://www.lydsy.com/JudgeOnline/problem.php?id=3098

BZOJ3099:http://www.lydsy.com/JudgeOnline/problem.php?id=3099

如果你仔细研究过了(或者至少仔细看过AC人数的话),我想你一定会明白字符串哈希的正确姿势的^_^

【字符串算法1】 再谈字符串Hash(优雅的暴力)的更多相关文章

  1. [VB.NET Tips]再谈字符串连接之内置池

    CLR自动维护一个称为"内置池"(暂存池)(intern pool)的表,在编译时此表包含程序中声明的每个唯一的字符串常量的单个实例,以及以编程方式创建的String类的任何唯一实 ...

  2. Python 再谈字符串

    字符串除了要用引号来创建之外,其他和元组一样,不能修改,如果要修改只能用切片或者拼接的方式. 其他的什么乱七八糟的运算符都一样 一些不同 capitalize()-将字符串的第一个字母大写 str1. ...

  3. 【字符串算法2】浅谈Manacher算法

    [字符串算法1] 字符串Hash(优雅的暴力) [字符串算法2]Manacher算法 [字符串算法3]KMP算法 这里将讲述  字符串算法2:Manacher算法 问题:给出字符串S(限制见后)求出最 ...

  4. 【字符串算法3】浅谈KMP算法

    [字符串算法1] 字符串Hash(优雅的暴力) [字符串算法2]Manacher算法 [字符串算法3]KMP算法 这里将讲述  [字符串算法3]KMP算法 Part1 理解KMP的精髓和思想 其实KM ...

  5. 转载:字符串hash总结(hash是一门优雅的暴力!)

    转载自:远航休息栈 字符串Hash总结 Hash是什么意思呢?某度翻译告诉我们: hash 英[hæʃ] 美[hæʃ]n. 剁碎的食物; #号; 蔬菜肉丁;vt. 把…弄乱; 切碎; 反复推敲; 搞糟 ...

  6. 浅谈字符串Hash

    浅谈字符串Hash 本篇随笔讲解Hash(散列表)的一个重要应用:字符串Hash. 关于Hash Hash是一种数据结构,叫做Hash表(哈希表),也叫散列表.关于Hash的实现,其实与离散化颇为类似 ...

  7. 解决AES算法CBC模式加密字符串后再解密出现乱码问题

    问题 在使用 AES CBC 模式加密字符串后,再进行解密,解密得到的字符串出现乱码情况,通常都是前几十个字节乱码: 复现 因为是使用部门 cgi AESEncryptUtil 库,找到问题后,在这里 ...

  8. 浅谈字符串哈希 By cellur925

    前言 蒟蒻最近在复习字符串算法...但正如之前所说,我OI太菜被关起来了,本蒟蒻只能从最简单的哈希入手了TAT.而别的dalao都在学习AC自动机/后缀数组等高到不知哪里去的算法qwq. 基本思想 映 ...

  9. [算法2-数组与字符串的查找与匹配] (.NET源码学习)

    [算法2-数组与字符串的查找与匹配] (.NET源码学习) 关键词:1. 数组查找(算法)   2. 字符串查找(算法)   3. C#中的String(源码)   4. 特性Attribute 与内 ...

随机推荐

  1. docker 端口映射错误解决方法

    今天搞了半天shipyard,在网页上打开时无法显示容器和镜像,最后发现是docker端口映射错误,由于防火墙未关闭: 4月 12 18:51:29 localhost firewalld[757]: ...

  2. 【小程序】<image>图片实现宽度100%时,高度自适应

    *.wxss样式设置 .img{ width:100% } *.wxml给<image>标签添加属性  mode="widthFix" <image class= ...

  3. UWP ListView 绑定 单击 选中项 颜色

    refer: https://www.cnblogs.com/lonelyxmas/p/7650259.html using System; using System.Collections.Gene ...

  4. 基于Boost库的HTTP Post函数

    两个函数的区别: 提交表单数据和提交文本数据 表单数据: request_stream << "Content-Type: application/x-www-form-urle ...

  5. 【亲测有效】Nodepad++/Sublime Text3中Python脚本运行出现语法错误:IndentationError: unindent does not match any outer indentation level解决策略

    我在开发游戏的时候,发现一个python脚本,本来都运行好好的,然后写了几行代码,而且也都确保每行都对齐了,但是运行的时候,却出现语法错误: IndentationError: unindent do ...

  6. Oracle实用地址

    1.详细安装教程 https://jingyan.baidu.com/article/3c48dd34be2a32e10be35881.html

  7. numpy 初识(二)

    针对 numpy.array(序列)的实例介绍 ndim 数组(矩阵)的维度 size 所有元素的和 数学运算(+, -) 元素个数一样,对应位置相减 加,减,乘,平方一个数,执行广播形式:即都减去一 ...

  8. 阿里云ECS服务器源配置

    前段时间领取了阿里云ECS免费试用6个月的福利,此处记录一下服务器源配置过程和服务器用户创建过程. 一.CentOS源配置 1.备份 mv /etc/yum.repos.d/CentOS-Base.r ...

  9. Unity EasyTouch官方案例学习

    一.代码检测手势事件 1. EasyTouch4.x 写法 首先要手动在 Hierarchy 窗口添加 EasyTouch 物体,以触摸(Touch)手势为例,代码如下: using UnityEng ...

  10. FPGA千兆位收发器选择指南

    选择合适的千兆位收发器(GT)是通信和实时处理领域尤其需要重点考虑的设计事项,但特定的市场领域可能会存在太多的标准.协议或使用模型.有时针对某一种应用就会涉及到好几种标准,为了选择最适合的千兆位收发器 ...