• 字符串哈希


    哈希做法是固定的:首先设一个进制数base,并设一个模数mod

    而哈希其实就是把一个数转化为一个值,这个值是base进制的,储存在哈希表中,注意一下在存入的时候取模一下即可,下文列举三种哈希方式。

    常用seed: 13331

    自然溢出

    ull hashs(char s[]){
        int len=strlen(s);
        ull ans=0;
        for (int i=0;i<len;i++)
            ans=ans*base+(ull)s[i];
        return ans&0x7fffffff;//?
    }
    

    双哈希

    最稳妥的办法是选择两个10^9级别的质数,只有模这两个数都相等才判断相等,但常数略大,代码相对难写,目前暂时没有办法卡掉这种写法(除了卡时间让它超时)

    ull mod1=19260817;
    ull mod2=19660813;
    ull hash1(char s[]){
        int len=strlen(s);
        ull ans=0;
        for (int i=0;i<len;i++)
            ans=(ans*base+(ull)s[i])%mod1;
        return ans;
    }
    ull hash2(char s[]){
        int len=strlen(s);
        ull ans=0;
        for (int i=0;i<len;i++)
            ans=(ans*base+(ull)s[i])%mod2;
        return ans;
    }
    

    大质数哈希

    ull mod=212370440130137957ll;//是质数
    ull hashs(char s[]){
        int len=strlen(s);
        ull ans=0;
        for (int i=0;i<len;i++)
            ans=(ans*base+(ull)s[i])%mod;
        return ans;
    }
    

    例题

    给定n个字符串,问有多少个不同的字符串

    #include <bits/stdc++.h>
    using namespace std;
    typedef unsigned long long ull;
    ull base=131;
    struct data {
        ull x,y;
    } a[10010];
    char s[10010];
    int n,ans=1;
    ull mod1=19260817;
    ull mod2=19660813;
    ull hash1(char s[]) {
        int len=strlen(s);
        ull ans=0;
        for (int i=0; i<len; i++)
            ans=(ans*base+(ull)s[i])%mod1;
        return ans;
    }
    ull hash2(char s[]) {
        int len=strlen(s);
        ull ans=0;
        for (int i=0; i<len; i++)
            ans=(ans*base+(ull)s[i])%mod2;
        return ans;
    }
    bool comp(data a,data b) {
        return a.x<b.x;
    }
    main() {
        scanf("%d",&n);
        for (int i=1; i<=n; i++) {
            scanf("%s",s);
            a[i].x=hash1(s);
            a[i].y=hash2(s);
        }
        sort(a+1,a+n+1,comp);
        for (int i=2; i<=n; i++)
            if (a[i].x!=a[i-1].x || a[i-1].y!=a[i].y)
                ans++;
        printf("%d
    ",ans);
    }
    
  • 相关阅读:
    js append()和appendChild()和insertBefore()的区别
    webpack打包工具简单案例
    Vue $ref 的用法
    Vue学习笔记-作用域插槽
    Vue学习笔记-插槽基本使用
    Vue学习笔记-父子通信案例
    Echarts案例-折线图
    Echarts案例-柱状图
    软件构造实验三-递归下降分析分析法
    软件构造实验二-拷贝一个c文件 将其中的关键字int替换成float
  • 原文地址:https://www.cnblogs.com/ucprer/p/13885352.html
Copyright © 2020-2023  润新知