[ex-kmp] HDU 2019 Multi-University Training Contest 5-string matching

string matching

Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 262144/262144 K (Java/Others)
Total Submission(s): 3131    Accepted Submission(s): 724


Problem Description
String matching is a common type of problem in computer science. One string matching problem is as following:

Given a string  s[0len1], please calculate the length of the longest common prefix of s[ilen1] and s[0len1] for each i>0.

I believe everyone can do it by brute force.
The pseudo code of the brute force approach is as the following:

<center> </center>

We are wondering, for any given string, what is the number of compare operations invoked if we use the above algorithm. Please tell us the answer before we attempt to run this algorithm.
 

 

Input
The first line contains an integer  T, denoting the number of test cases.
Each test case contains one string in a line consisting of printable ASCII characters except space.

1T30

* string length 106 for every string
 

 

Output
For each test, print an integer in one line indicating the number of compare operations invoked if we run the algorithm in the statement against the input string.
 

 

Sample Input
3 _Happy_New_Year_
ywwyww
zjczzzjczjczzzjc
 

 

Sample Output
17
7
32

题意:

给一个字符串s[0...len-1],计算它 s[i…len−1] 和 s[0…len−1] 最长公共前缀的长度之和,如果最后一个匹配的不是串中的最后一个,则要额外加1

思路:

用ex-kmp求解

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 const int amn=1e6+6;
 4 char a[amn],str1[amn],str2[amn];
 5 int pos[amn],ex[amn],alen,blen;
 6 void f(){
 7     pos[1]=alen;
 8     int x=1;
 9     while(str2[x]==str2[x+1]&&x+1<=blen)x++;
10     pos[2]=x-1;
11     int k=2;
12     for(int i=3;i<=alen;i++){
13         int p=k+pos[k]-1,le=pos[i-k+1];
14         if(i+le<p+1)pos[i]=le;
15         else{
16             int j=p-i+1;
17             if(j<0)j=0;
18             while(str2[j+1]==str2[i+j]&&i+j<=blen)j++;
19             pos[i]=j;
20             k=i;
21         }
22     }
23     x=1;
24     while(str1[x]==str2[x]&&x<=blen)x++;
25     ex[1]=x-1;
26     k=1;
27     for(int i=2;i<=alen;i++){
28         int p=k+ex[k]-1,le=pos[i-k+1];
29         if(i+le<p+1)ex[i]=le;
30         else{
31             int j=p-i+1;
32             if(j<0)j=0;
33             while(str2[j+1]==str1[i+j]&&i+j<=alen&&j<=blen)j++;
34             ex[i]=j;
35             k=i;
36         }
37     }
38 }
39 int main(){
40     int T;
41     long long ans;
42     scanf("%d",&T);
43     while(T--){
44         scanf("%s",a);
45         alen=blen=strlen(a);
46         strcpy(str1+1,a);
47         strcpy(str2+1,a);
48         f();
49         ans=0;
50         for(int i=2;i<=alen;i++){
51             if(alen-i+1>ex[i]){
52                 ans+=ex[i]+1;
53             }
54             else ans+=ex[i];
55         }
56         printf("%lld\n",ans);
57     }
58 }
59 /***
60 给一个字符串s[0...len-1],计算它 s[i…len−1] 和 s[0…len−1] 最长公共前缀的长度之和,如果最后一个匹配的不是串中的最后一个,则要额外加1
61 用扩展kmp求解
62 ***/

 

全部评论

相关推荐

不愿透露姓名的神秘牛友
2025-12-17 16:48
今天九点半到公司,我跟往常一样先扫了眼电脑,屁活儿没有。寻思着没事干,就去蹲了个厕所,回来摸出手机刷了会儿。结果老板刚好路过,拍了我一下说上班别玩手机,我吓得赶紧揣兜里。也就过了四十分钟吧,我的直属领导把我叫到小隔间,上来就给我一句:“你玩手机这事儿把老板惹毛了,说白了,你可以重新找工作了,等下&nbsp;HR&nbsp;会来跟你谈。”&nbsp;我当时脑子直接宕机,一句话都没憋出来。后面&nbsp;HR&nbsp;找我谈话,直属领导也在旁边。HR&nbsp;说我这毛病不是一次两次了,属于屡教不改,不光上班玩手机,还用公司电脑看论文、弄学校的事儿。我当时人都傻了,上班摸鱼是不对,可我都是闲得发慌的时候才摸啊!而且玩手机这事儿,从来没人跟我说过后果这么严重,更没人告诉我在公司学个习也算犯错!连一次口头提醒都没有,哪儿来的屡教不改啊?更让我膈应的是,昨天部门刚开了会,说四个实习生里留一个转正,让大家好好表现。结果今天我就因为玩手机被开了。但搞笑的是,开会前直属领导就把我叫去小会议室,明明白白告诉我:“转正这事儿你就别想了,你的学历达不到我们部门要求,当初招你进来也没打算给你这个机会。”合着我没入贵厂的眼是吧?可我都已经被排除在转正名单外了,摸个鱼至于直接把我开了吗?真的太离谱了!
rush$0522:转正名单没进,大概率本来就没打算留你
摸鱼被leader发现了...
点赞 评论 收藏
分享
2025-12-30 16:42
同济大学 C++
仁狂躁使者:哎呀,不用担心,我当时配环境配了两天,项目捋不清就问问导师能不能用ai,慢慢就清了,会好起来的
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务