【每日力扣】32. 最长有效括号 416. 分割等和子集

🔥 个人主页: 黑洞晓威

😀你不必等到非常厉害,才敢开始,你需要开始,才会变的非常厉害

32. 最长有效括号

给你一个只包含 '('')' 的字符串,找出最长有效(格式正确且连续)括号

子串

的长度。

示例 1:

复制代码
输入:s = "(()"
输出:2
解释:最长有效括号子串是 "()"

示例 2:

复制代码
输入:s = ")()())"
输出:4
解释:最长有效括号子串是 "()()"

示例 3:

复制代码
输入:s = ""
输出:0

解题思路

这个问题可以使用栈来解决。具体思路如下:

  1. 使用栈来存储括号的下标。
  2. 初始化栈并将-1入栈(用于处理特殊情况)。
  3. 遍历字符串,对于每个字符:
    • 如果遇到左括号'(',将其下标入栈。
    • 如果遇到右括号')',弹出栈顶元素(即栈中最后一个可以构成有效括号匹配的下标),并计算当前位置到上一个可以构成有效括号匹配的下标之间的距离,更新最长有效括号的长度。
  4. 返回最长有效括号的长度。

代码实现

java 复制代码
import java.util.Stack;

class Solution {
    public int longestValidParentheses(String s) {
        int maxLength = 0;
        Stack<Integer> stack = new Stack<>();
        stack.push(-1); // 初始放入-1,用于处理特殊情况

        for (int i = 0; i < s.length(); i++) {
            char c = s.charAt(i);
            
            if (c == '(') {
                // 左括号入栈
                stack.push(i);
            } else {
                // 右括号
                stack.pop();
                if (stack.isEmpty()) {
                    stack.push(i); // 更新新的起点
                } else {
                    maxLength = Math.max(maxLength, i - stack.peek());
                }
            }
        }

        return maxLength;
    }
}

416. 分割等和子集

给你一个 只包含正整数非空 数组 nums 。请你判断是否可以将这个数组分割成两个子集,使得两个子集的元素和相等。

示例 1:

复制代码
输入:nums = [1,5,11,5]
输出:true
解释:数组可以分割成 [1, 5, 5] 和 [11] 。

示例 2:

复制代码
输入:nums = [1,2,3,5]
输出:false
解释:数组不能分割成两个元素和相等的子集。

解题思路

  1. 计算整个数组的元素和 total
  2. 如果 total 是奇数,直接返回 false,因为无法平分奇数的总和。
  3. 将问题转化为在数组中寻找一个子集,使其和等于 total / 2,也就是寻找一个子集的和等于总和的一半,在背包问题中可以看作是背包容量为 total / 2 的背包能否恰好装满。
  4. 使用动态规划解决背包问题,创建一个布尔类型的二维数组 dp,其中 dp[i][j] 表示在前 i 个元素中能否凑出和为 j
  5. 动态规划状态转移方程为 dp[i][j] = dp[i - 1][j] || dp[i - 1][j - nums[i - 1]],即当前元素可以选择放入或不放入背包。
  6. 最后返回 dp[nums.length][total / 2] 的值。

代码实现

java 复制代码
class Solution {
    public boolean canPartition(int[] nums) {
        int total = 0;
        for (int num : nums) {
            total += num;
        }
        
        if (total % 2 != 0) {
            return false;
        }
        
        int target = total / 2;
        boolean[][] dp = new boolean[nums.length + 1][target + 1];
        dp[0][0] = true;
        
        for (int i = 1; i <= nums.length; i++) {
            dp[i][0] = true; // 可以凑出和为0
            for (int j = 1; j <= target; j++) {
                if (j >= nums[i - 1]) {
                    dp[i][j] = dp[i - 1][j] || dp[i - 1][j - nums[i - 1]];
                } else {
                    dp[i][j] = dp[i - 1][j];
                }
            }
        }
        
        return dp[nums.length][target];
    }
}
相关推荐
黑色的山岗在沉睡34 分钟前
LeetCode 494. 目标和
算法·leetcode·职场和发展
haoly19894 小时前
数据结构和算法篇-线性查找优化-移至开头策略
数据结构·算法·移至开头策略
莫叫石榴姐7 小时前
SQL百题斩:从入门到精通,一站式解锁数据世界
大数据·数据仓库·sql·面试·职场和发展
学Linux的语莫7 小时前
机器学习数据处理
java·算法·机器学习
earthzhang20217 小时前
【1007】计算(a+b)×c的值
c语言·开发语言·数据结构·算法·青少年编程
你总是一副不开心的样子(´ . .̫ .7 小时前
一、十天速通Java面试(第三天)
java·面试·职场和发展·java面试
2301_803554529 小时前
C++联合体(Union)详解:与结构体的区别、联系与深度解析
java·c++·算法
sali-tec10 小时前
C# 基于halcon的视觉工作流-章42-手动识别文本
开发语言·人工智能·算法·计算机视觉·c#·ocr
SandySY10 小时前
品三国谈人性
算法·架构
小欣加油10 小时前
leetcode 62 不同路径
c++·算法·leetcode·职场和发展