《蓝桥杯每日一题》KMP算法·AcWing 141. 周期

简介: 《蓝桥杯每日一题》KMP算法·AcWing 141. 周期

1.题目描述


一个字符串的前缀是从第一个字符开始的连续若干个字符,例如 abaab 共有 55 个前缀,分别是 aababaabaaabaab


我们希望知道一个 N 位字符串 S 的前缀是否具有循环节。


换言之,对于每一个从头开始的长度为 i(i>1)的前缀,是否由重复出现的子串 A 组成,即 AAA…A (A 重复出现 K 次,K>1)。


如果存在,请找出最短的循环节对应的 K 值(也就是这个前缀串的所有可能重复节中,最大的 K值)。


输入格式

输入包括多组测试数据,每组测试数据包括两行。


第一行输入字符串 S 的长度 N。


第二行输入字符串 S。


输入数据以只包括一个 0 的行作为结尾。


输出格式


对于每组测试数据,第一行输出 Test case # 和测试数据的编号。

接下来的每一行,输出具有循环节的前缀的长度 i 和其对应 K,中间用一个空格隔开。

前缀长度需要升序排列

在每组测试数据的最后输出一个空行。


数据范围


2≤N≤1000000


输入样例:


3aaa4abcd12aabaabaabaab0


输出样例:


Test case #12 23 3Test case #2Test case #32 26 29 312 4


2.思路解析


不清楚kmp的可以先看下这篇

https://blog.csdn.net/m0_68055637/article/details/128596380


首先发现一个性质,对字符串S自己匹配求出next数组,然后根据定义,对于每一个i,s[i-next[i]+1~i]与s[1~next[i]]是相等的,而且不存在更大的next值满足条件.


既然如此的话,那么我们发现当i-next[i]能够整除i时候,那么s[1~i-next[i]]就是s[i-1]的最小循环元,至于次数那么也就是i/(i-next[i]).


3.代码


import java.util.Scanner;
public class Main {
    public static void main(String[] args) {
        Scanner sc=new Scanner(System.in);
        num=0;
        while(true){
            int n=sc.nextInt();
            if(n==0){
                break;
            }
            String str1=sc.next();
            num++;
            System.out.println("Test case #"+num);
            shuchu(n,str1);
            System.out.println();
        }
    }
    public static int num;
    public static void shuchu(int n,String str){
        //KMP的next的数组
        int[]next=new int[n];
        //此处next的数组是使用最大长度表实现 跟next[0]=-1意义不同
        next[0]=0;
        for (int i = 1,j=0; i <n ; i++) {
            while(j>0 &&str.charAt(j)!=str.charAt(i)){
                j= next[j-1];
            }
            if(str.charAt(i)==str.charAt(j)){
                j++;
            }
            next[i]=j;
            if(next[i]!=0){
           //i+1是当前字符串的长度 i+1-next[i]是这个字符串重复的子串的长度
                if((i+1)%(i+1-next[i])==0){ 
                    int temp=(i+1)/(i+1-next[i]); //个数
                    System.out.println((i+1)+" "+temp);
                }
            }
        }
    }
}


感谢你能看完, 如有错误欢迎评论指正,有好的思路可以交流一波,如果对你有帮助的话,点个赞支持下

相关文章
|
28天前
|
算法 搜索推荐 Java
【潜意识Java】深度解析黑马项目《苍穹外卖》与蓝桥杯算法的结合问题
本文探讨了如何将算法学习与实际项目相结合,以提升编程竞赛中的解题能力。通过《苍穹外卖》项目,介绍了订单配送路径规划(基于动态规划解决旅行商问题)和商品推荐系统(基于贪心算法)。这些实例不仅展示了算法在实际业务中的应用,还帮助读者更好地准备蓝桥杯等编程竞赛。结合具体代码实现和解析,文章详细说明了如何运用算法优化项目功能,提高解决问题的能力。
58 6
|
28天前
|
算法 Java C++
【潜意识Java】蓝桥杯算法有关的动态规划求解背包问题
本文介绍了经典的0/1背包问题及其动态规划解法。
48 5
|
4月前
|
算法 测试技术 C++
【动态规划算法】蓝桥杯填充问题(C/C++)
【动态规划算法】蓝桥杯填充问题(C/C++)
|
4月前
|
存储 机器学习/深度学习 算法
蓝桥杯练习题(三):Python组之算法训练提高综合五十题
蓝桥杯Python编程练习题的集合,涵盖了从基础到提高的多个算法题目及其解答。
163 3
蓝桥杯练习题(三):Python组之算法训练提高综合五十题
|
4月前
|
算法
第四章 KMP算法理论基础
第四章 KMP算法理论基础
40 0
|
4月前
|
算法
KMP算法
KMP算法
54 0
|
6月前
|
算法 C++
A : DS串应用–KMP算法
这篇文章提供了KMP算法的C++实现,包括计算模式串的next数组和在主串中查找模式串位置的函数,用于演示KMP算法的基本应用。
|
6月前
|
算法
KMP算法
KMP算法
48 0
|
2天前
|
算法 数据安全/隐私保护 计算机视觉
基于FPGA的图像双线性插值算法verilog实现,包括tb测试文件和MATLAB辅助验证
本项目展示了256×256图像通过双线性插值放大至512×512的效果,无水印展示。使用Matlab 2022a和Vivado 2019.2开发,提供完整代码及详细中文注释、操作视频。核心程序实现图像缩放,并在Matlab中验证效果。双线性插值算法通过FPGA高效实现图像缩放,确保质量。
|
1月前
|
算法 数据安全/隐私保护 计算机视觉
基于Retinex算法的图像去雾matlab仿真
本项目展示了基于Retinex算法的图像去雾技术。完整程序运行效果无水印,使用Matlab2022a开发。核心代码包含详细中文注释和操作步骤视频。Retinex理论由Edwin Land提出,旨在分离图像的光照和反射分量,增强图像对比度、颜色和细节,尤其在雾天条件下表现优异,有效解决图像去雾问题。