• CH 1401


    题目链接:传送门

    描述

    很久很久以前,森林里住着一群兔子。有一天,兔子们想要研究自己的 DNA 序列。我们首先选取一个好长好长的 DNA 序列(小兔子是外星生物,DNA 序列可能包含 26 个小写英文字母),然后我们每次选择两个区间,询问如果用两个区间里的 DNA 序列分别生产出来两只兔子,这两个兔子是否一模一样。注意两个兔子一模一样只可能是他们的 DNA 序列一模一样。

    输入格式

    第一行一个 DNA 字符串 S。
    接下来一个数字 m,表示 m 次询问。
    接下来 m 行,每行四个数字 l1, r1, l2, r2,分别表示此次询问的两个区间,注意字符串的位置从1开始编号。
    其中 1 ≤ length(S), m ≤ 1000000

    输出格式

    对于每次询问,输出一行表示结果。如果两只兔子完全相同输出 Yes,否则输出 No(注意大小写)

    样例输入

    aabbaabb
    3
    1 3 5 7
    1 3 6 8
    1 2 1 2

    样例输出

    Yes
    No
    Yes

    来源

    罗翔宇,北京大学2014年数据结构与算法A(实验班)期末考试

    题解:

    参考《算法竞赛进阶指南》P62-64。

    给所有可能出现的字符赋值,例如所有 $a$ 到 $z$ 小写字母赋值 $1$ 到 $26$,然后给定一个远大于所有字符集的大小的数字 $P$,把所有字符串看成是一个  $P$ 进制数。

    然后在给定一个模数 $M$,所有字符串转成 $P$ 进制数后,再去模这个 $M$,得到的结果即为哈希函数值。

    假设 $S,T$ 是两个字符串,而 $c$ 是一个字符,则有

    $egin{array}{l} Hleft( {S + c} ight) = left( {Hleft( S ight) imes P + Hleft( c ight)} ight)mod M \ Hleft( {S + T} ight) = left( {Hleft( S ight) imes P^{lenleft( T ight)} + Hleft( T ight)} ight)mod M \ end{array}$

    根据以上两个公式,可以 $O(len(S))$ 处理一个字符串的所有前缀子串的哈希值,同时可以 $O(1)$ 的查询任意子串的哈希值。

    AC代码:

    #include<bits/stdc++.h>
    using namespace std;
    typedef unsigned long long ull;
    
    const int P=131;
    const int maxn=1000000+10;
    
    char s[maxn];
    int q;
    
    ull H[maxn],Ppow[maxn];
    void pretreat(int len)
    {
        H[0]=0;
        Ppow[0]=1;
        for(int i=1;i<=len;i++)
        {
            H[i]=H[i-1]*P+(s[i]-'a'+1);
            Ppow[i]=Ppow[i-1]*P;
        }
    }
    
    int main()
    {
        scanf("%s",s+1);
        pretreat(strlen(s+1));
        scanf("%d",&q);
        for(int i=1;i<=q;i++)
        {
            int l1,r1,l2,r2;
            scanf("%d%d%d%d",&l1,&r1,&l2,&r2);
            ull A=H[r1]-H[l1-1]*Ppow[r1-(l1-1)];
            ull B=H[r2]-H[l2-1]*Ppow[r2-(l2-1)];
            if(A==B) printf("Yes
    ");
            else printf("No
    ");
        }
    }

    时间复杂度:$O(len(S) + m)$

  • 相关阅读:
    异地协作,A地上传jar包到B地服务器上传速率慢
    linux一行命令查杀进程
    maven项目创建.m2文件夹
    模态框传递参数
    测试身份证信息
    jenkins:邮件配置良心之作
    python:不错的python编程核心思想
    jenkins:忘记密码怎么办
    docker:如何查看容器的挂载目录
    JavaScript + PHP 实现刷新继续保持倒计时的按钮
  • 原文地址:https://www.cnblogs.com/dilthey/p/9693840.html
Copyright © 2020-2023  润新知