2024.1.9力扣每日一题——字符串中的额外

2024.1.9

      • 题目来源
      • 我的题解
        • [方法一 集合+动态规划](#方法一 集合+动态规划)
        • [方法二 字典树+动态规划](#方法二 字典树+动态规划)

题目来源

力扣每日一题;题序:2707

我的题解

方法一 集合+动态规划

题目的要求是讲字符串s划分为多个不重叠的子串,然后尽可能使得更多的子串匹配到dictionary中的字符串,然后将这些匹配的子串删除之后,使得剩下的额外字符最小。

可以假设s的长度为n,分割的方式:

  • 把s的最后一个字符s[n-1]当做是额外字符,这问题转化为求长度为n-1的子问题;
  • 找到一个s的后缀s[j...n-1]构成的子串在dictionary中,这问题转换为j-1的子问题。

因此,可以将得出动态规划的转移方程:

d p [ i ] = d p [ i − 1 ] + 1 , s [ i − 1 ] 当做额外字符 dp[i]=dp[i-1]+1,s[i-1]当做额外字符 dp[i]=dp[i−1]+1,s[i−1]当做额外字符

d p [ i ] = m i n ( d p [ i ] , d p [ j ] ) , s 的后缀 s [ j . . . n − 1 ] 构成的子串在 d i c t i o n a r y 中 dp[i]=min(dp[i],dp[j]),s的后缀s[j...n-1]构成的子串在dictionary中 dp[i]=min(dp[i],dp[j]),s的后缀s[j...n−1]构成的子串在dictionary中

为了判断子串是不是在dictionary中,将dictionary中的字符串使用HashSet集合存储。
时间复杂度 :O( n 3 + m n^3+m n3+m)。n为s的长度,m为dictionary的长度。m是初始化HashSet的时间; n 3 n^3 n3分别是递推需要O(n),每次子问题需要O(n),并且求子串需要O(n)。
空间复杂度:O(n+m)。dp数组的空间和HashSet的空间。

java 复制代码
public int minExtraChar(String s, String[] dictionary) {
	int n=s.length();
   int m=dictionary.length;
   int[] dp=new int[n+1];
   Arrays.fill(dp,Integer.MAX_VALUE);
   Set<String> set=new HashSet<>();
   for(int i=0;i<m;i++){
       set.add(dictionary[i]);
   }
   dp[0]=0;
   for(int i=1;i<=n;i++){
   		// s[i-1]当做额外字符
       dp[i]=dp[i-1]+1;
       for(int j=i-1;j>=0;j--){
       		//s的后缀s[j...n-1]构成的子串在dictionary中
           if(set.contains(s.substring(j,i))){
               dp[i]=Math.min(dp[i],dp[j]);
           }
       }
   }
   return dp[n];
}
方法二 字典树+动态规划

在方法一中查找某个子串是否在dictionary中效率较低,因为查找了s[i+1,j],再次查找s[i,j],这时后者仅仅第一个字符不同,但是仍需要从开始开始对比,要花费O(n)的时间重复查找。

可以使用字典树存储dictionary中字符串的逆序,这样在字典树上找到表示后缀s[i+1,j]的节点后,只需要再花O(1)的时间来判断表示后缀s[i,j]的节点是否存在。
时间复杂度:O( n 2 + ∣ S ∣ n^2+|S| n2+∣S∣)。|S|表示dictionary中最长字符串的长度

空间复杂度:O(n+∣T∣⋅Σ)。|T|表示dictionary中所有字符串的长度之和,Σ为字符集的大小,这里是26。

java 复制代码
 public int minExtraChar(String s, String[] dictionary) {
int n=s.length();
    int m=dictionary.length;
    int[] dp=new int[n+1];
    Arrays.fill(dp,Integer.MAX_VALUE);
    Trie t=new Trie();
    // 将dictionary中的字符串逆序插入字典树
    for(int i=0;i<m;i++){
        StringBuilder sb=new StringBuilder(dictionary[i]);
        t.insert(sb.reverse().toString());
    }
    dp[0]=0;
    for(int i=1;i<=n;i++){
        dp[i]=dp[i-1]+1;
        Trie node=t;
        for(int j=i-1;j>=0;j--){
        // 若当前节点为空表示在dictionary中找不到s[j,i]的字符串
            if(node!=null){
            // 看下一个字典树节点是不是存在
                node=node.searchSuffix(s.charAt(j));
                if(node!=null&&node.isEnd()){
                    dp[i]=Math.min(dp[i],dp[j]);
                }
            }
        }
    }
    return dp[n];
}
class Trie {
    Trie children[];//子节点
    boolean isEndOf;//是否是结尾
    public Trie() {
        children=new Trie[26];
        isEndOf=false;
    }
    //插入新的字符串
    public void insert(String word) {
        Trie cur=this;
        for(int i=0;i<word.length();i++){
            int ch=word.charAt(i)-'a';
            if(cur.children[ch]==null){//若没有该分支,则新建分支
                cur.children[ch]=new Trie();
            }
            cur=cur.children[ch];
        }
        cur.isEndOf=true;//记录结尾
    }
    //判断当前字典树节点是不是一个结尾
    public boolean isEnd(){
        return isEndOf;
    }
    private Trie searchSuffix(char ch){//查找ch为结尾的字典树节点
        Trie cur=this;
        if(cur==null||cur.children[ch-'a']==null){
            return null;
        }
        cur=cur.children[ch-'a'];
        return cur;
    }
}

有任何问题,欢迎评论区交流,欢迎评论区提供其它解题思路(代码),也可以点个赞支持一下作者哈😄~

相关推荐
雷神乐乐8 分钟前
File.separator与File.separatorChar的区别
java·路径分隔符
小刘|12 分钟前
《Java 实现希尔排序:原理剖析与代码详解》
java·算法·排序算法
jjyangyou17 分钟前
物联网核心安全系列——物联网安全需求
物联网·算法·安全·嵌入式·产品经理·硬件·产品设计
逊嘘31 分钟前
【Java语言】抽象类与接口
java·开发语言·jvm
van叶~34 分钟前
算法妙妙屋-------1.递归的深邃回响:二叉树的奇妙剪枝
c++·算法
简简单单做算法35 分钟前
基于Retinex算法的图像去雾matlab仿真
算法·matlab·图像去雾·retinex
morris13139 分钟前
【SpringBoot】Xss的常见攻击方式与防御手段
java·spring boot·xss·csp
云卓SKYDROID1 小时前
除草机器人算法以及技术详解!
算法·机器人·科普·高科技·云卓科技·算法技术
七星静香1 小时前
laravel chunkById 分块查询 使用时的问题
java·前端·laravel
Jacob程序员1 小时前
java导出word文件(手绘)
java·开发语言·word