hdu 3374 String Problem (kmp+最大最小表示法)

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3374

题目大意:输出最大和最小的是从哪一位开始的,同时输出最小循环节的个数。

这里简单介绍对字符串最小表示的方法:

(1)  利用两个指针p1, p2。初始化时p1指向s[0], p2指向s[1]。

(2)  k = 0开始,检验s[p1+k] 与 s[p2+k] 对应的字符是否相等,如果相等则k++,一直下去,直到找到第一个不同,(若k试了一个字符串的长度也没找到不同,则那个位置就是最小表示位置,算法终止并返回)。则该过程中,s[p1+k] 与 s[p2+k]的大小关系,有三种情况:

(A). s[p1+k] > s[p2+k],则p1滑动到p1+k+1处 --- 即s1[p1->p1+k]不会是该循环字符串的“最小表示”的前缀。 k置为0

(B). s[p1+k] < s[p2+k],则p2滑动到p2+k+1处, k置为0

(C). s[p1+k] = s[p2+k],则 k++; if (k == len) 返回结果。

注:这里滑动方式有个小细节,若滑动后p1 == p2,将正在变化的那个指针再+1。直到p1、p2把整个字符串都检验完毕,返回两者中小于 len 的值。

(3)   如果 k == len, 则返回p1与p2中的最小值

最大和最小表示完全类似,简单的改变一下即可。下面详见代码。

 1 #include <iostream>
 2 #include <cstdio>
 3 #include <cstring>
 4 using namespace std;
 5 int next[1000010],len,t;
 6 char ch[1000010];
 7
 8 void get_next()
 9 {
10     int i=0,j=-1;
11     next[0]=-1;
12     while (i<len)
13     {
14         if (j==-1||ch[i]==ch[j])
15         {
16             i++;
17             j++;
18             next[i]=j;
19         }
20         else
21             j=next[j];
22     }
23 }
24
25 int Min_(char s[],int len)
26 {
27     int p1=0,p2=1,k=0;
28     while (p1<len&&p2<len&&k<len)
29     {
30         t=s[(p1+k)%len]-s[(p2+k)%len];
31         if (t==0)
32             k++;
33         else
34         {
35             if (t>0)
36                 p1=p1+k+1;
37             else
38                 p2=p2+k+1;
39             if (p1==p2)
40                 p2++;
41             k=0;
42         }
43     }
44     return min(p1,p2);
45 }
46
47 int Max_(char s[],int len)
48 {
49     int p1=0,p2=1,k=0;
50     while (p1<len&&p2<len&&k<len)
51     {
52         t=s[(p1+k)%len]-s[(p2+k)%len];
53         if (t==0)
54             k++;
55         else
56         {
57             if (t>0)
58                 p2=p2+k+1;
59             else
60                 p1=p1+k+1;
61             if (p1==p2)
62                 p2++;
63             k=0;
64
65         }
66     }
67     return p1<p2?p1:p2;
68 }
69
70 int main()
71 {
72     while(~scanf("%s",ch))
73     {
74         len=strlen(ch);
75         get_next();
76         int Max=Max_(ch,len);
77         int Min=Min_(ch,len);
78         //cout<<Max<<Min<<endl;
79         int sum=0;
80         if(len%(len-next[len])==0)
81             sum=len/(len-next[len]);
82         else
83             sum=1;
84         printf ("%d %d %d %d\n",Min+1,sum,Max+1,sum);
85     }
86     return 0;
87 }

hdu 3374 String Problem (kmp+最大最小表示法)

时间: 2024-12-09 16:43:15

hdu 3374 String Problem (kmp+最大最小表示法)的相关文章

HDU 3374 String Problem (KMP+最大最小表示)

KMP,在有循环节的前提下: 循环节 t = len-next[len], 个数num = len/(len-next[len]); 个人理解,如果有循环节,循环节长度必定小于等于len/2, 换句话说next[len]>=len/2; 对于len%(len-next)!=0的这种情况不讨论,循环节不存在. 下面是假设循环节存在的情况 当次数等于2, 对于abcabc这种情况就不用说了,len = 6, next[len] = 3; 当次数大于2, 对于串a1 a2 a3 a4 a5 a6 a7

HDU 3374 String Problem(最大最小表示)

Description: Give you a string with length N, you can generate N strings by left shifts. For example let consider the string “SKYLONG”, we can generate seven strings: String Rank SKYLONG 1 KYLONGS 2 YLONGSK 3 LONGSKY 4 ONGSKYL 5 NGSKYLO 6 GSKYLON 7 a

HDU 3374 String Problem (KMP+最大最小表示)

转载请注明出处,谢谢http://blog.csdn.net/acm_cxlove/article/details/7854526       by---cxlove 题目:输出最大最小表示是从哪一位开始,而且输出数量 http://acm.hdu.edu.cn/showproblem.php?pid=3374 数量好求,肯定是字符串的循环节,循环节可以直接通过KMP的Next数组得到.这个和yobobobo曾经研究过半天.. 对于最大最小表示法,就是将字符串不断旋转,得到字典序最大或者最小的.

hdu 3374 String Problem

String Problem http://acm.hdu.edu.cn/showproblem.php?pid=3374 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 3153    Accepted Submission(s): 1272 Problem Description Give you a string with leng

HDOJ3374 String Problem [KMP最小循环节点]+[最小(大)表示法]

String Problem Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 1442    Accepted Submission(s): 645 Problem Description Give you a string with length N, you can generate N strings by left shifts

hdu 3746 Cyclic Nacklace (KMP求最小循环节)

//len-next[len]为最小循环节的长度 # include <stdio.h> # include <algorithm> # include <string.h> using namespace std; int len; char a[100010]; int next[100010]; void Getnext() { int i=0,j=-1; next[0]=-1; while(i<=len) { if(j==-1||a[i]==a[j]) i

hdu 5772 String problem(最大权闭合图)

题目链接:hdu 5772 String problem 题意: 给你一个字符串,只含有数字. 你需要选择出一个子序列,使得这个子序列的权值最大. 这个子序列如果这个数字第一次出现就ans-=bx,否则就-=ax 然后如果第i个字符和第j个字符都在子序列里面,那么ans+=w[i][j] 问你最大ans是多少 官方题解: 1 #include<bits/stdc++.h> 2 #define F(i,a,b) for(int i=a;i<=b;++i) 3 using namespace

HDU 3374 String Proble

String Problem Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 3183    Accepted Submission(s): 1295 Problem Description Give you a string with length N, you can generate N strings by left shifts

hdu P3374 String Problem

今天又在lyk大佬的博客学会了——最小表示法(异常激动发篇题解纪念一下说在前面:给luogu提个建议最小表示法的题太少了,都被hdu抢去了!!! 我们先看一下题目 看完后可以用一个字概括——蒙,两个字——懵逼 在这里我提供题目大意: 输出最大和最小的是从哪一位开始的,同时输出最小循环节的个数. 由于本人懒于写字符串最小表示法,那么我们就来借鉴一下lykkk的优秀总结 看完之后,显然我们就明白了许多 因为题目中让我们同时求出最大和最小的起始位置 所以我们不仅要来一遍最小表示法,还要来一遍最大表示法