单词拆分/分割等和子集

1.单词拆分

二维dp的思路还是比较复杂(目前看)

java 复制代码
class Solution {
    public boolean wordBreak(String s, List<String> wordDict) {
        Set<String> dict = new HashSet<>(wordDict);

        int n = s.length();
        //以下标i结尾的是否可以频出
        boolean[][] dp = new boolean[n][n];

        for(int i =0;i<n;i++){
            for(int j = i;j<n;j++){
                if(dict.contains(s.substring(i,j+1))){//左实右虚
                    dp[i][j] = true;
                }
            }
        }

//1.二维dp,先枚举长度
    for(int len = 1;len<=n;len++){

        for(int i = 0;i<=n-len;i++){//2.起始坐标
            int j = i+len-1; //3.终点坐标

            if(dp[i][j]){
                continue;//跳过本次循环,i++

            }
            for(int k =i;k<j;k++){
                //5.选择一种可以把这块,用已有字典切分的方案就行
                if(dp[i][k]&&dp[k+1][j]){
                    dp[i][j] =true;
                    break;//就知道了i,j 可以切分, 跳出循环
                }
            }
        }
    }


        return dp[0][n-1];
    }
}

2.分割等和子集

java 复制代码
class Solution {
    public boolean canPartition(int[] nums) {
        //这里【有点类似背包问题
        //dp[i][j] 选择前i个数,能否和为j

        //
        int sum = 0;
        for(int i = 0;i<nums.length;i++){
            sum+= nums[i];
        }
        if(sum%2 !=0){
            return false;
        }
        int target = sum/2;
        boolean[][] dp = new boolean[nums.length][target+1];
        dp[0][0] = true;
        for(int i =0;i<nums.length;i++){
            dp[i][0] = true;

        }
        for(int i = 1;i<nums.length;i++){
            for(int j = 1;j<=target;j++){
                if(nums[i]>j){
                    dp[i][j] = dp[i-1][j];
                }else{
                    dp[i][j] = dp[i-1][j] || dp[i-1][j-nums[i]];//选或者不选
                }
            }
        }
        return dp[nums.length-1][target];
    }
}

上述代码需要下面的补充,同时注意选择第0个元素的时候初始化,<= 这样选择是为了可以状态转移递推过去

即使小于在计算的时候,如(1,2)也会被偶数限制条件return;(8,2,2,2)就是无法划分的就是false初始化的时候

java 复制代码
// 初始化第0行
dp[0][0] = true;
if(nums[0] <= target){
    dp[0][nums[0]] = true;  // ⭐ 选第0个元素
}

// 初始化第0列
for(int i = 1; i < nums.length; i++){
    dp[i][0] = true;
}

3.最长有效括号(这个耗得时间长些)

java 复制代码
class Solution {
    public int longestValidParentheses(String s) {
        //dp[i] 表示以i坐标为结尾的最长有效括号字串长度
        int maxLen = 0;
        int pre = 0;
        char[] chars = s.toCharArray();
        int[] dp = new int[s.length()];

        for(int i =1;i<s.length();i++){
            if(chars[i] == ')'){
                //推测上一个'('可能的位置
                pre = i - dp[i-1] -1;

                            if(pre>0 && chars[pre]=='('){
            dp[i] = dp[i-1] + 2 + dp[pre-1];
            }
             if(pre==0 && chars[pre]=='('){
            dp[i] = dp[i-1] + 2 ;
            }

            }


            maxLen =Math.max(maxLen,dp[i]);
        } 
        return maxLen;
    }
}
相关推荐
智者知已应修善业18 小时前
【51单片机独立按键控制数码管自增自减】2023-10-5
c++·经验分享·笔记·算法·51单片机
凯瑟琳.奥古斯特18 小时前
页面置换算法详解与对比
开发语言·分布式·职场和发展
2301_8008951018 小时前
第十四届蓝桥杯国赛b组真题---备战国赛版h
算法·蓝桥杯·深度优先
生信之灵18 小时前
告别模板配准:LAMNr Flow如何用一次求逆破解多模态解剖对齐难题
人工智能·算法
焜昱错眩..18 小时前
力扣周赛难题 3906.统计网格路径中好整数的数目——自我拆解学习与分析(数位dp上下界的奇妙)
学习·算法·leetcode·动态规划
wangl_9218 小时前
初探 C# 15 的 Union Types
java·开发语言·算法·c#·.net·.net core
Smile灬凉城66618 小时前
逻辑回归数据集
算法·机器学习·逻辑回归
笑虾18 小时前
dotnet 8 实现 XXTEA 解密核心算法
算法
龙佚18 小时前
噪声抑制技术:让语音更清晰
算法·架构