【每日力扣】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];
    }
}
相关推荐
hqyjzsb8 小时前
企业AI人才库的搭建体系与长效运营管理方案
人工智能·学习·职场和发展·创业创新·学习方法·业界资讯·改行学it
舟舟亢亢8 小时前
算法总结——二叉树【hot100】(上)
java·开发语言·算法
weixin_477271699 小时前
根象:树根。基石。基于马王堆帛书《周易》原文及甲骨文还原周朝生活活动现象(《函谷门》原创)
算法·图搜索算法
普通网友10 小时前
多协议网络库设计
开发语言·c++·算法
努力努力再努力wz10 小时前
【Linux网络系列】:TCP 的秩序与策略:揭秘传输层如何从不可靠的网络中构建绝对可靠的通信信道
java·linux·开发语言·数据结构·c++·python·算法
daxi15010 小时前
C语言从入门到进阶——第9讲:函数递归
c语言·开发语言·c++·算法·蓝桥杯
持续学习的程序员+111 小时前
强化学习Q-chunking算法
算法
我命由我1234511 小时前
Photoshop - Photoshop 工具栏(67)修补工具
学习·ui·职场和发展·求职招聘·职场发展·学习方法·photoshop
Polaris北11 小时前
第二十七天打卡
开发语言·c++·算法
风吹乱了我的头发~12 小时前
Day30:2026年2月20日打卡
算法