[toc]ios
# 如下内容做废,太多错误了,等我有时间重写算法
说一下什么是Hash,说白了就是把一大坨字符用一些神奇的数来表示,能够说是把字符加密了.函数
简单一点就是一个像函数同样的东西,你放进去一个值,它给你输出来一个值。输出的值就是Hash值。通常Hash值会比原来的值更好储存(更小)或比较。加密
字符串hash的灵魂就是尽可能让不一样的字符串对应惟一的hsah的值 .而要实现这一效果就要选对方法不然就咕咕咕了spa
举个栗子:code
若是咱们的加密方法是把字符的ascal加起来,那就咕咕咕了.xml
好比:blog
ababa字符串
babaaget
加起来是同样的,咕咕咕....
因此应该怎么hash?
它的主要思路是选取恰当的进制,能够把字符串中的字符当作一个大数字中的每一位数字,不过比较字符串和比较大数字的复杂度并无什么区别(高精数的比较也是O(n)O(n)的),但只要把它对一个数取模,而后认为取模后的结果相等原数就相等,那么就能够在必定的错误率的基础上O(1)O(1)进行判断了。
那么咱们选择什么进制比较好?
首先不要把任意字符对应到数字0,好比假如把a对应到数字0,那么将不能只从Hash结果上区分ab和b(虽然能够额外判断字符串长度,但不把任意字符对应到数字0更加省事且没有任何反作用),通常而言,把a-z对应到数字1-26比较合适。
关于进制的选择实际上很是自由,大于全部字符对应的数字的最大值,不要含有模数的质因子(那还模什么),好比一个字符集是a到z的题目,选择2七、23三、19260817都是能够的。
模数的选择(尽可能仍是要选择质数):
绝大多数状况下,不要选择一个109109级别的数,由于这样随机数据都会有Hash冲突,根据生日悖论,随便找上109−−−√109个串就有大几率出现至少一对Hash 值相等的串
最稳妥的办法是选择两个109109级别的质数,只有模这两个数都相等才判断相等,但常数略大,代码相对难写,目前暂时没有办法卡掉这种写法(除了卡时间让它超时)
若是能背过或在考场上找出一个10181018级别的质数(Miller-Rabin),也相对靠谱,主要用于前一种担忧会超时,后一种担忧被卡。
偷懒的写法就是直接使用unsigned long long,不手动进行取模,它溢出时会自动对264
送上一道模板题:
如题,给定N个字符串(第i个字符串长度为Mi,字符串内包含数字、大小写字母,大小写敏感),请求出N个字符串中共有多少个不一样的字符串。
输入格式:
第一行包含一个整数N,为字符串的个数。
接下来N行每行包含一个字符串,为所提供的字符串。
输出格式:
输出包含一行,包含一个整数,为不一样的字符串个数。
时空限制:1000ms,128M
数据规模:
对于30%的数据:N<=10,Mi≈6,Mmax<=15;
对于70%的数据:N<=1000,Mi≈100,Mmax<=150
对于100%的数据:N<=10000,Mi≈1000,Mmax<=1500
代码:
#include<iostream> #include<cstdio> #include<algorithm> #include<cmath> #include<queue> #include<stack> #include<vector> #include<map> #include<string> #include<cstring> #define ll long long int #define mod 1000000007 using namespace std; const long long int maxn=99999999999999; const int minn=-999999999; long long base=131; long long a[15010]; char s[15010]; int n,ans=1; long long hash(char s[]) { int len=strlen(s); long long ans=0; for (int i=0; i<len; i++) ans=ans*base+(long long )s[i]; return ans%maxn; } int main() { scanf("%d",&n); for (int i=1; i<=n; i++) { scanf("%s",s); a[i]=hash(s); } sort(a+1,a+n+1); for (int i=2; i<=n; i++) if (a[i]!=a[i-1]) ans++; printf("%d\n",ans); }
送上取膜数对答案的影响:
.
改了一下mod为0x3f3f3f就