【每日一题】466. 统计重复个数-2024.1.2

题目:

466. 统计重复个数

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

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

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

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

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

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

示例 1:

复制代码
输入:s1 = "acb", n1 = 4, s2 = "ab", n2 = 2
输出:2

示例 2:

复制代码
输入:s1 = "acb", n1 = 1, s2 = "acb", n2 = 1
输出:1

提示:

  • 1 <= s1.length, s2.length <= 100
  • s1s2 由小写英文字母组成
  • 1 <= n1, n2 <= 106

解答:

代码:

复制代码
class Solution {
    public int getMaxRepetitions(String s1, int n1, String s2, int n2) {
        if(n1==0){
            return 0;
        }
        int s1cnt=0,index=0,s2cnt=0;
        // recall 是我们用来找循环节的变量,它是一个哈希映射
        // 我们如何找循环节?假设我们遍历了 s1cnt 个 s1,此时匹配到了第 s2cnt 个 s2 中的第 index 个字符
        // 如果我们之前遍历了 s1cnt' 个 s1 时,匹配到的是第 s2cnt' 个 s2 中同样的第 index 个字符,那么就有循环节了
        // 我们用 (s1cnt', s2cnt', index) 和 (s1cnt, s2cnt, index) 表示两次包含相同 index 的匹配结果
        // 那么哈希映射中的键就是 index,值就是 (s1cnt', s2cnt') 这个二元组
        // 循环节就是;
        //    - 前 s1cnt' 个 s1 包含了 s2cnt' 个 s2
        //    - 以后的每 (s1cnt - s1cnt') 个 s1 包含了 (s2cnt - s2cnt') 个 s2
        // 那么还会剩下 (n1 - s1cnt') % (s1cnt - s1cnt') 个 s1, 我们对这些与 s2 进行暴力匹配
        // 注意 s2 要从第 index 个字符开始匹配
        Map<Integer,int[]> recall=new HashMap<Integer,int[]>();
        int[] preLoop=new int[2];
        int[] inLoop=new int[2];
        while(true){
             // 我们多遍历一个 s1,看看能不能找到循环节
             ++s1cnt;
             for(int i=0;i<s1.length();i++){
                 char ch=s1.charAt(i);
                 if(ch==s2.charAt(index)){
                     index+=1;
                     if(index==s2.length()){
                         ++s2cnt;
                         index=0;
                     }
                 }
             }
             //还没有找到循环节,所有的 s1 就用完了
             if(s1cnt==n1){
                 return s2cnt/n2;
             }
             //出现了之前的 index,表示找到了循环节
             if(recall.containsKey(index)){
                 int[] value=recall.get(index);
                 int s1cntPrime=value[0];
                 int s2cntPrime=value[1];
                 // 前 s1cnt' 个 s1 包含了 s2cnt' 个 s2
                 preLoop=new int[]{s1cntPrime,s2cntPrime};
                 // 以后的每 (s1cnt - s1cnt') 个 s1 包含了 (s2cnt - s2cnt') 个 s2
                 inLoop=new int[]{s1cnt-s1cntPrime,s2cnt-s2cntPrime};
                 break;
             }else{
                 recall.put(index,new int[]{s1cnt,s2cnt});
             }
        }
        // ans 存储的是 S1 包含的 s2 的数量,考虑的之前的 preLoop 和 inLoop
        int ans=preLoop[1]+(n1-preLoop[0])/inLoop[0]*inLoop[1];
        // S1 的末尾还剩下一些 s1,我们暴力进行匹配
        int rest=(n1-preLoop[0])%inLoop[0];
        for(int i=0;i<rest;i++){
            for(int j=0;j<s1.length();j++){
                char ch=s1.charAt(j);
                if(ch==s2.charAt(index)){
                    index++;
                    if(index==s2.length()){
                        ans++;
                        index=0;
                    }
                }
            }
        }
        // S1 包含 ans 个 s2,那么就包含 ans / n2 个 S2
        return ans/n2;
    }
}

结果:

相关推荐
B站_计算机毕业设计之家35 分钟前
预测算法:股票数据分析预测系统 股票预测 股价预测 Arima预测算法(时间序列预测算法) Flask 框架 大数据(源码)✅
python·算法·机器学习·数据分析·flask·股票·预测
想唱rap2 小时前
C++ list 类的使用
c语言·开发语言·数据结构·c++·笔记·算法·list
l1t2 小时前
利用DuckDB SQL求解集合数学题
数据库·sql·算法·集合·duckdb
yuyanjingtao2 小时前
CCF-GESP 等级考试 2024年9月认证C++四级真题解析
c++·算法·青少年编程·gesp·csp-j/s
微笑尅乐2 小时前
洗牌算法讲解——力扣384.打乱数组
算法·leetcode·职场和发展
Lei_3359672 小时前
[算法]背包DP(01背包、完全背包问题、多重背包、分组背包、混合背包问题、有依赖的背包问题等)
c++·算法
uesowys2 小时前
华为OD算法开发指导-比赛的冠亚季军
算法·华为od
天选之女wow2 小时前
【代码随想录算法训练营——Day48】单调栈——42.接雨水、84.柱状图中最大的矩形
算法·leetcode
不知名。。。。。。。。2 小时前
算法之动态规划
算法·动态规划
lingchen19062 小时前
MATLAB图形绘制基础(一)二维图形
开发语言·算法·matlab