【dp力扣】环绕字符串中唯一的子字符串

题目链接

题目:

java 复制代码
class Solution {
    public int findSubstringInWraproundString(String s) {

    }
}

审题:

给定base这个字符串,它是"abcd......xyz"围成的一个环形无线长度的字符串(只有小写的26个字母)。

在给定一个字符串s,题目要求计算:在s串中的所有的子串(至少包含一个字符)中,有多少个子串在base串中出现过

示例:

细节注意:

在示例二中,我们要知道重复出现的字符我们只记录一次!!!

尽管'cac'中两个c都存在于base串中,但是只算一个。

状态表示:

dpi

i表示,以0,i为范围,i位置为结尾,所有子数组在base串中出现的总次数。

java 复制代码
class Solution {
    public int findSubstringInWraproundString(String s) {
         
            
             //把字符串变成字符数组,等一下好操作。
             char[] sArr=s.toCharArray(); 
 
                   int n=sArr.length;


            //dp[i]表示,以[0,i]为范围,i位置为结尾,所有子数组在base串中出现的总次数。
            int[] dp=new int[n];

          
    }
}

通过dp计算正确答案(先讲)

这里我们先不推导状态方程,暂且假设dp数组已经被我们填表完成了。

依据这个dp数组,如何求出答案所需要的子数组出现的总次数呢?

一些童鞋可能会直接说dp数组求和即可。按照经验的确如此,刚开始我也是这样想的。

但,事出反常必有妖。

当前这个问题是解决这个算法的重要环节,所以我放到这里讲。

刚才小编在示例处强调过,重复的字符只能计算一次!所以如果直接对dp求和,那么结果只能是>=正确答案(因为加上了重复的次数)。

比如,s="abdb",在dp数组中dp1dp和dp3(dpi含义,请看上面代码注释)两个值一定有重复的计数,因为"ab"和"abdb"中都有b这个字符。因此如果是简单的dp求和,结果可能就会比正确答案大。

如何避免重复计数的情况发生呢?(注意下面的内容第一次听可能有点绕,但不难,细心听几遍就懂了)

可以使用哈希的思想避免重复计数。

题目提示说了,s和base一样都是小写的字母,所以定义一个哈希表:

java 复制代码
class Solution {
    public int findSubstringInWraproundString(String s) {
         
            
             //把字符串变成字符数组,等一下好操作。
             char[] sArr=s.toCharArray(); 
 
                   int n=sArr.length;


            //dp[i]表示,以[0,i]为范围,i位置为结尾,所有子数组在base串中出现的总次数。
            int[] dp=new int[n];

            这里假定dp数组已经填表完毕了!!



            //确定返回值(用于去重)
            int[] hash=new int[26];

 
    }
}

把dp表映射到hash表中,hash求和就是答案了(映射前hash的每一个元素都是0)。

那么dp表怎么映射到hash表中呢?

实际上,借助字符数组sArr即可:

hash0对应的是sArri=='a'

hash1对应的是sArr\[j=='b'

所以:

hash0=sArri

hash1=sArrj

不过,以上这种模式并没有实现去重。

如果sArri和sArrj都等于'b',怎么办?

这实际上就回到了刚才的"abdb"中,dp1和dp3出现重复计数的问题了。

其实很简单,直接取最大值:max(dp1,dp3)即可。

为什么呢?

dpi表示,以0,i为范围,i位置为结尾,所有子数组在base串中出现的总次数。

而dp3实际上已经包含了子数组dp1中的所有情况,

也就是说如果i<j,si==sj,那么一定有:dpi<dpj!(细细理解这句话)

java 复制代码
class Solution {
    public int findSubstringInWraproundString(String s) {
         
            
             //把字符串变成字符数组,等一下好操作。
             char[] sArr=s.toCharArray(); 
 
                   int n=sArr.length;


            //dp[i]表示,以[0,i]为范围,i位置为结尾,所有子数组在base串中出现的总次数。
            int[] dp=new int[n];

            这里假定dp数组已经填表完毕了!!

    

            //确定返回值
            int[] hash=new int[26];

                //开始把dp映射到hash(用于去重)
            for(int i=0;i<n;i++){
                hash[sArr[i]-'a']=Math.max(hash[sArr[i]-'a'],dp[i]);
            }

            //hash求和就是答案
            int sum=0;
            for(int h:hash){
                sum+=h;
            }

        return sum;
    }
}

状态方程的初始化以及推导

在回顾以下状态表示dpii表示,以0,i为范围,i位置为结尾,所有子数组在base串中出现的总次数。

所以dpi和dpi-1有什么关系呢?

倘若两个字符si和si-1是连续的,即si-1+1!=si(ASCII码)并且si-1!='z'且si!='a'。那么dpi=dpi-1+1。(以i为结尾,实际上,也就增加了si这一个字母)

反之,如果不满足粗体条件,dpi=1。

如下AC代码:

java 复制代码
class Solution {
    public int findSubstringInWraproundString(String s) {
         
            
             //把字符串变成字符数组,等一下好操作。
             char[] sArr=s.toCharArray(); 
 
                   int n=sArr.length;


            //dp[i]表示,以[0,i]为范围,i位置为结尾,所有子数组在base串中出现的总次数。
            int[] dp=new int[n];

            这里假定dp数组已经填表完毕了!!

            dp[0]=1;//涉及i-1,只需要把这个初始化即可!!!!
            for(int i=1;i<n;i++){
                if(sArr[i-1]+1==sArr[i]||(sArr[i-1]=='z'&&sArr[i]=='a')){
                        dp[i]=dp[i-1]+1;
                    }
                else dp[i]=1;
            }

            //确定返回值
            int[] hash=new int[26];

                //开始把dp映射到hash(用于去重)
            for(int i=0;i<n;i++){
                hash[sArr[i]-'a']=Math.max(hash[sArr[i]-'a'],dp[i]);
            }

            //hash求和就是答案
            int sum=0;
            for(int h:hash){
                sum+=h;
            }

        return sum;
    }
}
相关推荐
七牛开发者14 小时前
MCP 到底是什么?为什么 Agent 都想接上它
算法·aigc·agent
kisshyshy21 小时前
从递归到迭代,一文吃透二叉树的核心知识与 JavaScript 实现
javascript·算法·代码规范
To_OC1 天前
LC 49 字母异位词分组:想到哈希表很简单,选对 key 才是精髓
javascript·算法·leetcode
用户938515635072 天前
从 O(n²) 到 O(nlogn):一文读懂快速排序的“快”与“妙”
javascript·算法
To_OC2 天前
手写快排次次翻车?别死背快排模板了,这才是面试官想听的底层逻辑
javascript·算法·排序算法
饼干哥哥2 天前
Reddit VOC调研太慢?搭一个AI专家团队半小时洞察任何品类|以猫用饮水机为例
人工智能·算法·ai编程
地平线开发者2 天前
Transformer模型部署之性能优化指南
算法
地平线开发者2 天前
人在途中:从“编译失败”到“模型可落地”——CUDA 自定义算子
算法·自动驾驶
半个落月2 天前
从递归到快速排序:用 JavaScript 把分治思想讲明白
javascript·算法·面试