【LeetCode 热题100】DP 实战进阶:最长递增子序列、乘积最大子数组、分割等和子集(力扣300 / 152/ 416 )(Go语言版)

简介: 本文深入解析三道经典的动态规划问题:**最长递增子序列(LIS)**、**乘积最大子数组** 和 **分割等和子集**。 - **300. LIS** 通过 `dp[i]` 表示以第 `i` 个元素结尾的最长递增子序列长度,支持 O(n²) 动态规划与 O(n log n) 的二分优化。 - **152. 乘积最大子数组** 利用正负数特性,同时维护最大值与最小值的状态转移方程。 - **416. 分割等和子集** 转化为 0-1 背包问题,通过布尔型 DP 实现子集和判断。 总结对比了三题的状态定义与解法技巧,并延伸至相关变种问题,助你掌握动态规划的核心思想与灵活应用!

🧠 DP 实战进阶:最长递增子序列、乘积最大子数组、分割等和子集(LeetCode 300 / 152 / 416)

在动态规划的学习路径中,这三道题常被视作进阶经典,它们分别对应不同的状态定义与优化思路:

  • 📈 300. 最长递增子序列(LIS):子序列类动态规划
  • ✖️ 152. 乘积最大子数组:带正负号波动的区间 DP
  • 🎯 416. 分割等和子集:0-1 背包问题的变种

📈 一、300. 最长递增子序列

📌 题目描述

给你一个整数数组 nums,返回其中最长严格递增子序列的长度。


💡 解题思路(一):动态规划

状态定义:

  • dp[i] 表示以 nums[i] 结尾的最长递增子序列长度。

状态转移方程:

for j := 0; j < i; j++ {
   
    if nums[i] > nums[j] {
   
        dp[i] = max(dp[i], dp[j] + 1)
    }
}

时间复杂度:O(n²)


✅ Go 实现

func lengthOfLIS(nums []int) int {
   
    n := len(nums)
    dp := make([]int, n)
    for i := range dp {
   
        dp[i] = 1
    }

    maxLen := 1
    for i := 1; i < n; i++ {
   
        for j := 0; j < i; j++ {
   
            if nums[i] > nums[j] {
   
                dp[i] = max(dp[i], dp[j]+1)
            }
        }
        maxLen = max(maxLen, dp[i])
    }
    return maxLen
}

func max(a, b int) int {
   
    if a > b {
   
        return a
    }
    return b
}

💡 解法(二):二分优化(贪心 + 二分)

维护一个数组 tailstails[i] 表示长度为 i+1 的递增子序列中末尾最小的值。

时间复杂度:O(n log n)


✖️ 二、152. 乘积最大子数组

📌 题目描述

给你一个整数数组 nums,找出一个乘积最大的连续子数组,返回该乘积。


💡 解题思路

因为负负得正的存在,我们需要同时记录当前的最大值和最小值:

  • dpMax[i]:以 i 结尾的最大乘积
  • dpMin[i]:以 i 结尾的最小乘积

状态转移:

dpMax[i] = max(nums[i], nums[i]*dpMax[i-1], nums[i]*dpMin[i-1])
dpMin[i] = min(nums[i], nums[i]*dpMax[i-1], nums[i]*dpMin[i-1])

✅ Go 实现(空间优化)

func maxProduct(nums []int) int {
   
    maxVal, minVal := nums[0], nums[0]
    res := nums[0]

    for i := 1; i < len(nums); i++ {
   
        tempMax := maxVal
        maxVal = max(nums[i], max(nums[i]*maxVal, nums[i]*minVal))
        minVal = min(nums[i], min(nums[i]*tempMax, nums[i]*minVal))
        res = max(res, maxVal)
    }
    return res
}

🎯 三、416. 分割等和子集

📌 题目描述

给定一个只包含正整数的非空数组,判断能否将其分成两个子集,使得两个子集的和相等


💡 解题思路

转换为 0-1 背包问题

  • 目标是找到一个子集,使其和为 sum/2
  • 状态定义:dp[j] 表示是否存在和为 j 的子集;
  • 状态转移:
dp[j] = dp[j] || dp[j - num]

✅ Go 实现

func canPartition(nums []int) bool {
   
    total := 0
    for _, num := range nums {
   
        total += num
    }
    if total%2 != 0 {
   
        return false
    }
    target := total / 2
    dp := make([]bool, target+1)
    dp[0] = true

    for _, num := range nums {
   
        for j := target; j >= num; j-- {
   
            dp[j] = dp[j] || dp[j-num]
        }
    }
    return dp[target]
}

🔚 总结与对比

题目 类型 状态定义 难点与技巧
300. 最长递增子序列 子序列 DP dp[i] 为以 i 结尾 LIS 长度 可二分优化
152. 乘积最大子数组 区间 DP 同时维护最大最小 处理正负数乘积
416. 分割等和子集 0-1 背包 dp[j] 为是否能组成 j 典型集合和划分

📘 写在最后

这三道题展示了动态规划在处理不同问题时的状态建模能力转移逻辑多样性。可以举一反三,比如:

  • LIS 拓展题:673. 最长递增子序列的个数
  • 背包拓展题:494. 目标和
  • 区间动态规划:42. 接雨水、91. 解码方法

目录
相关文章
|
3月前
|
Linux Go iOS开发
Go语言100个实战案例-进阶与部署篇:使用Go打包生成可执行文件
本文详解Go语言打包与跨平台编译技巧,涵盖`go build`命令、多平台构建、二进制优化及资源嵌入(embed),助你将项目编译为无依赖的独立可执行文件,轻松实现高效分发与部署。
|
4月前
|
数据采集 数据挖掘 测试技术
Go与Python爬虫实战对比:从开发效率到性能瓶颈的深度解析
本文对比了Python与Go在爬虫开发中的特点。Python凭借Scrapy等框架在开发效率和易用性上占优,适合快速开发与中小型项目;而Go凭借高并发和高性能优势,适用于大规模、长期运行的爬虫服务。文章通过代码示例和性能测试,分析了两者在并发能力、错误处理、部署维护等方面的差异,并探讨了未来融合发展的趋势。
360 0
|
4月前
|
存储 人工智能 Go
Go-Zero全流程实战即时通讯
Go-Zero 是一个功能丰富的微服务框架,适用于开发高性能的即时通讯应用。它具备中间件、工具库和代码生成器,简化开发流程。本文介绍其环境搭建、项目初始化及即时通讯功能实现,涵盖用户认证、消息收发和实时推送,帮助开发者快速上手。
327 0
|
3月前
|
存储 前端开发 JavaScript
Go语言实战案例-项目实战篇:编写一个轻量级在线聊天室
本文介绍如何用Go语言从零实现一个轻量级在线聊天室,基于WebSocket实现实时通信,支持多人消息广播。涵盖前后端开发、技术选型与功能扩展,助你掌握Go高并发与实时通信核心技术。
|
2月前
|
存储 人工智能 算法
从零掌握贪心算法Java版:LeetCode 10题实战解析(上)
在算法世界里,有一种思想如同生活中的"见好就收"——每次做出当前看来最优的选择,寄希望于通过局部最优达成全局最优。这种思想就是贪心算法,它以其简洁高效的特点,成为解决最优问题的利器。今天我们就来系统学习贪心算法的核心思想,并通过10道LeetCode经典题目实战演练,带你掌握这种"步步为营"的解题思维。
|
4月前
|
负载均衡 监控 Java
微服务稳定性三板斧:熔断、限流与负载均衡全面解析(附 Hystrix-Go 实战代码)
在微服务架构中,高可用与稳定性至关重要。本文详解熔断、限流与负载均衡三大关键技术,结合API网关与Hystrix-Go实战,帮助构建健壮、弹性的微服务系统。
520 1
微服务稳定性三板斧:熔断、限流与负载均衡全面解析(附 Hystrix-Go 实战代码)
|
4月前
|
安全 Go 开发者
Go语言实战案例:使用sync.Mutex实现资源加锁
在Go语言并发编程中,数据共享可能导致竞态条件,使用 `sync.Mutex` 可以有效避免这一问题。本文详细介绍了互斥锁的基本概念、加锁原理及实战应用,通过构建并发安全的计数器演示了加锁与未加锁的区别,并封装了一个线程安全的计数器结构。同时对比了Go中常见的同步机制,帮助开发者理解何时应使用 `Mutex` 及其注意事项。掌握 `Mutex` 是实现高效、安全并发编程的重要基础。
|
4月前
|
数据采集 Go API
Go语言实战案例:使用context控制协程取消
本文详解 Go 语言中 `context` 包的使用,通过实际案例演示如何利用 `context` 控制协程的生命周期,实现任务取消、超时控制及优雅退出,提升并发程序的稳定性与资源管理能力。
|
4月前
|
数据采集 Go API
Go语言实战案例:多协程并发下载网页内容
本文是《Go语言100个实战案例 · 网络与并发篇》第6篇,讲解如何使用 Goroutine 和 Channel 实现多协程并发抓取网页内容,提升网络请求效率。通过实战掌握高并发编程技巧,构建爬虫、内容聚合器等工具,涵盖 WaitGroup、超时控制、错误处理等核心知识点。
|
4月前
|
Go 开发者
Go语言实战案例:使用select监听多个channel
本文为《Go语言100个实战案例 · 网络与并发篇》第5篇,详解Go并发核心工具`select`的使用。通过实际案例讲解如何监听多个Channel、实现多任务处理、超时控制和非阻塞通信,帮助开发者掌握Go并发编程中的多路异步事件处理技巧。

热门文章

最新文章