LeetCode 466. 统计重复个数,循环字符串匹配优化

2024-01-03 15:04

本文主要是介绍LeetCode 466. 统计重复个数,循环字符串匹配优化,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、题目

1、题目描述

定义 str = [s, n] 表示 str 由 n 个字符串 s 连接构成。

  • 例如,str == ["abc", 3] =="abcabcabc" 。

如果可以从 s2 中删除某些字符使其变为 s1,则称字符串 s1 可以从字符串 s2 获得。

  • 例如,根据定义,s1 = "abc" 可以从 s2 = "abdbec" 获得,仅需要删除加粗且用斜体标识的字符。

现在给你两个字符串 s1 和 s2 和两个整数 n1 和 n2 。由此构造得到两个字符串,其中 str1 = [s1, n1]str2 = [s2, n2] 。

请你找出一个最大整数 m ,以满足 str = [str2, m] 可以从 str1 获得。

2、接口描述

class Solution {
public:int getMaxRepetitions(string s1, int n1, string s2, int n2) {}
};

3、原题链接

466. 统计重复个数


二、解题报告

1、思路分析

官方给出的解法是循环节解法,笔者采用了类似KMP优化字符串匹配的思想,不难想到我们只要计算出n1个s1中匹配的s2的数量然后除以n2就是答案,那么我们无非就是在s1 * n1这个字符串上进行了一次字符串匹配,我们暴力的求解,即从第0个字符一直到len1 * n1个字符进行和s2的匹配,记录s2的数目则有如下暴力代码:

class Solution {
public:Solution(){ios::sync_with_stdio(false);cin.tie(0),cout.tie(0);}int getMaxRepetitions(string s1, int n1, string s2, int n2) {int len1 = s1.size() , len2 = s2.size() , p2 = 0 , cnt = 0;for(int i = 0 ; i < n1 ; i++)for(int j = 0 ; j < len1 ; j++)if(s1[j] == s2[p2]){p2++;if(p2 == len2)p2 = 0 , cnt ++;}return cnt / n2;}
};

n1可以达到1e6量级,len1可以达到100量级,那么总体就是1e8量级,不出意外,49个样例全部通过但是卡常,所以还是没过。

但这说明只要对暴力解法进行小优化就可以通过本题。由于在一个循环字符串中寻找另一个字符串的数目,这显然有许多的冗余匹配,那么我们不妨像kmp那样,开一个数组nxt[],nxt[i]存储从s2下标i开始和单个s1能够匹配的字符串数目(这里s2是循环的,但是统计的是和单个s1匹配的字符数目),nxt的求解可以暴力求解,由于len1,len2不超过100,所以整体预处理不超过1e4

有了nxt后,我们再在n1 * s1 上进行匹配就可以优化为O(n1)了,为什么呢?

两个字符串从0开始匹配,第一个s1匹配结束,s2下标从0变为(nxt[0] + 0)%len2,再跟第二个s1匹配,s2下标又变为 ((0 + nxt[0]) % len1 + nxt[(0 + nxt[0]) % len1]) % len1……

直接看代码会明了很多

2、复杂度

时间复杂度: O(n1) 空间复杂度:O(L)

3、代码详解

​相比于循环节的方法,该方法的代码要简洁很多,当然该方法也可以进一步优化为循环节方法,如果让你将该方法进一步优化,那么阁下又该如何应对呢?
class Solution {
public:
int nxt[101] , len1 , len2 , sum;int getMaxRepetitions(string s1, int n1, string s2, int n2) {memset(nxt , 0 , sizeof(nxt));len1 = s1.size() , len2 = s2.size() , sum = 0;for(int i = 0 ; i < len2 ; i++)for(int j = i , k = 0 ; k < len1 ; k++)if(s2[j] == s1[k])nxt[i]++ , j = (j + 1) % len2;for(int i = 0 ,j = 0 ; i < n1 ; i++)sum += nxt[j] , j = (j + nxt[j]) % len2;return sum / (len2 * n2);}
};

这篇关于LeetCode 466. 统计重复个数,循环字符串匹配优化的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/566091

相关文章

Vue3 的 shallowRef 和 shallowReactive:优化性能

大家对 Vue3 的 ref 和 reactive 都很熟悉,那么对 shallowRef 和 shallowReactive 是否了解呢? 在编程和数据结构中,“shallow”(浅层)通常指对数据结构的最外层进行操作,而不递归地处理其内部或嵌套的数据。这种处理方式关注的是数据结构的第一层属性或元素,而忽略更深层次的嵌套内容。 1. 浅层与深层的对比 1.1 浅层(Shallow) 定义

哈希leetcode-1

目录 1前言 2.例题  2.1两数之和 2.2判断是否互为字符重排 2.3存在重复元素1 2.4存在重复元素2 2.5字母异位词分组 1前言 哈希表主要是适合于快速查找某个元素(O(1)) 当我们要频繁的查找某个元素,第一哈希表O(1),第二,二分O(log n) 一般可以分为语言自带的容器哈希和用数组模拟的简易哈希。 最简单的比如数组模拟字符存储,只要开26个c

HDFS—存储优化(纠删码)

纠删码原理 HDFS 默认情况下,一个文件有3个副本,这样提高了数据的可靠性,但也带来了2倍的冗余开销。 Hadoop3.x 引入了纠删码,采用计算的方式,可以节省约50%左右的存储空间。 此种方式节约了空间,但是会增加 cpu 的计算。 纠删码策略是给具体一个路径设置。所有往此路径下存储的文件,都会执行此策略。 默认只开启对 RS-6-3-1024k

好题——hdu2522(小数问题:求1/n的第一个循环节)

好喜欢这题,第一次做小数问题,一开始真心没思路,然后参考了网上的一些资料。 知识点***********************************无限不循环小数即无理数,不能写作两整数之比*****************************(一开始没想到,小学没学好) 此题1/n肯定是一个有限循环小数,了解这些后就能做此题了。 按照除法的机制,用一个函数表示出来就可以了,代码如下

hdu1496(用hash思想统计数目)

作为一个刚学hash的孩子,感觉这道题目很不错,灵活的运用的数组的下标。 解题步骤:如果用常规方法解,那么时间复杂度为O(n^4),肯定会超时,然后参考了网上的解题方法,将等式分成两个部分,a*x1^2+b*x2^2和c*x3^2+d*x4^2, 各自作为数组的下标,如果两部分相加为0,则满足等式; 代码如下: #include<iostream>#include<algorithm

使用opencv优化图片(画面变清晰)

文章目录 需求影响照片清晰度的因素 实现降噪测试代码 锐化空间锐化Unsharp Masking频率域锐化对比测试 对比度增强常用算法对比测试 需求 对图像进行优化,使其看起来更清晰,同时保持尺寸不变,通常涉及到图像处理技术如锐化、降噪、对比度增强等 影响照片清晰度的因素 影响照片清晰度的因素有很多,主要可以从以下几个方面来分析 1. 拍摄设备 相机传感器:相机传

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

poj2406(连续重复子串)

题意:判断串s是不是str^n,求str的最大长度。 解题思路:kmp可解,后缀数组的倍增算法超时。next[i]表示在第i位匹配失败后,自动跳转到next[i],所以1到next[n]这个串 等于 n-next[n]+1到n这个串。 代码如下; #include<iostream>#include<algorithm>#include<stdio.h>#include<math.

poj3261(可重复k次的最长子串)

题意:可重复k次的最长子串 解题思路:求所有区间[x,x+k-1]中的最小值的最大值。求sa时间复杂度Nlog(N),求最值时间复杂度N*N,但实际复杂度很低。题目数据也比较水,不然估计过不了。 代码入下: #include<iostream>#include<algorithm>#include<stdio.h>#include<math.h>#include<cstring

spoj705( 求不相同的子串个数)

题意:求串s的不同子串的个数 解题思路:任何子串都是某个后缀的前缀,对n个后缀排序,求某个后缀的前缀的个数,减去height[i](第i个后缀与第i-1 个后缀有相同的height[i]个前缀)。 代码如下: #include<iostream>#include<algorithm>#include<stdio.h>#include<math.h>#include<cstrin