zoukankan      html  css  js  c++  java
  • HDU1711Number Sequence (kmp找母串ns[]中子串ms[]第一次出现时,首位的位置)

    Number Sequence
    Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)
    Total Submission(s): 13207    Accepted Submission(s): 5979
    Problem Description
    Given two sequences of numbers : a[1], a[2], ...... , a[N], and b[1], b[2], ...... , b[M] (1 <= M <= 10000, 1 <= N <= 1000000). Your task is to find a number K which make a[K] = b[1], a[K + 1] = b[2], ...... , a[K + M - 1] = b[M]. If there are more than one K exist, output the smallest one.
     
    Input
    The first line of input is a number T which indicate the number of cases. Each case contains three lines. The first line is two numbers N and M (1 <= M <= 10000, 1 <= N <= 1000000). The second line contains N integers which indicate a[1], a[2], ...... , a[N]. The third line contains M integers which indicate b[1], b[2], ...... , b[M]. All integers are in the range of [-1000000, 1000000].
     
    Output
    For each test case, you should output one line which only contain K described above. If no such K exists, output -1 instead.
     
    Sample Input

    2 13 5 1 2 1 2 3 1 2 3 1 3 2 1 2 1 2 3 1 3 13 5 1 2 1 2 3 1 2 3 1 3 2 1 2 1 2 3 2 1

     
    Sample Output

    6 -1


     
    Source
    HDU 2007-Spring Programming Contest
     
    Recommend
    lcy   |   We have carefully selected several similar problems for you:  3336 3746 1251 2222 1867 

    题意:找到母串ns[]中子串ms[]第一次出现时,首位的位置

    思路:kmp

    注意:

    1.getnext()带入的一定是子串ms[]
    2.题目为数组,与字符串不同,注意定义类型     

    #include <iostream>
    #include <cstdio>
    #include <cstring>
    #include <cstdlib>
    #include <algorithm>
    using namespace std;
    int t,n,m;
    int ns[1000005];
    int ms[10005];
    int nexts[1000005];
    
    void getnext()
    {
        int i=0;
        int k=-1;
        nexts[0]=-1;
        while(i<m)
        {
            if(k==-1||ms[i]==ms[k])
            {
                i++;
                k++;
                nexts[i]=k;
            }
            else
                k=nexts[k];
        }
    }
    void get_ex()
    {
        int j=0,ans=-1;
        int k=0;
        getnext();
        // printf("%d
    ",m);
        while(k<n)
        {
            if(j==-1||ns[k]==ms[j])
            {
                j++;
                k++;
            }
            else
                j=nexts[j];
            if(j==m)
            {
                // printf("%d
    ",k);
                ans=k;
                break;
            }
        }
        //  printf("%d
    ",m);
        if(ans!=-1)
            printf("%d
    ",ans-m+1);
        else
            printf("-1
    ");
    }
    int main()
    {
        scanf("%d",&t);
        while(t--)
        {
            scanf("%d%d",&n,&m);
            for(int i=0; i<n; i++)
                scanf("%d",&ns[i]);
            for(int j=0; j<m; j++)
                scanf("%d",&ms[j]);
            get_ex();
        }
        return 0;
    }

    下面是next数组的预处理核心代码:

    复制代码
    void find_next_() {
        int j=0,i=1;
        next_[1]=0;
        while(i<m) {
            if(j==0||b[j]==b[i]) {
                ++i,++j;
                if(b[i]!=b[j])next_[i]=j;
                else next_[i]=next_[j];
            }
            else j = next_[j];
        }
    }
    复制代码

    这回是以下图为例来进行示范:(因为上面的例子不太好用)

    我们使用KMP预处理对B数组进行一个搜索,然后得到一个next数组,实际上就是统计出B中相同的连续元素,然后进行一个标记的操作

    然后我们经过处理后得到的next数组为:我们先不要管这个数组为什么要通过计算,以及这个数组的作用是什么,我们再来模拟一遍KMP的操作,因为时间原因,我就简单操作一遍,我们直接跳到不同的那个步骤:

    经过我们观察,是不是能够发现B[6]之前有着相同的元素1,2,1,那么我们是不是能够让相同的部分重叠,来减少这个匹配所移动的次数呢?

    你看这样是不是能一步达到我们所需要的条件,然后我们继续操作,

     你看,我们我们又节省了一部分时间,但是没有1 2 1可以匹配了怎么办?没关系,接着往下看,

    这样我们又将两个1匹配在了一起,但是A[i]!=B[j],并且之前只有一个元素可以匹配,完全不能移动了,那么我们应该怎么继续操作呢?什么,B[ ]整体向右移动?没错!这样我们又能得到,

    哇,所有匹配都全部完成了诶!你看,这样是不是很能节省我们之前那种操作循环所需要的时间?其实KMP当中的预处理就是扮演了一个这样的角色,他告诉我们当A[i]!=B[j]时我们应该怎样整体移动B[ ]的数组,next预处理函数所做的只是查找当A[i]与B[j]不匹配时,A[i]之前的后m个元素与B[ ]数组开头的前m个元素顺序相同时的提取下标的操作,你看看之前的操作是不是这样进行操作的呢?

  • 相关阅读:
    mongodb 4.X 创建用户以及授权
    文件上传之后,MD5不一致,大小一致
    golang macaron各种形式入参封装
    说说非托管资源的回收
    layui数据表格批量删除
    除按钮外禁用所有表单项
    could not read Username for 'https://github.com': No error
    进入Web的殿堂与Django的初接触
    shell脚本中的小数运算
    js遍历数组时删除元素最终结果不对
  • 原文地址:https://www.cnblogs.com/dshn/p/4750737.html
Copyright © 2011-2022 走看看