Gemini永久会员 Go 返回最长有效子串长度

Go 返回最长有效子串长度

以下是几种在 Go 中实现查找最长有效括号子串长度的方法:

方法一:动态规划

go 复制代码
func longestValidParentheses(s string) int {
    n := len(s)
    if n == 0 {
        return 0
    }
    
    dp := make([]int, n)
    maxLen := 0
    
    for i := 1; i < n; i++ {
        if s[i] == ')' {
            if s[i-1] == '(' {
                if i >= 2 {
                    dp[i] = dp[i-2] + 2
                } else {
                    dp[i] = 2
                }
            } else {
                prev := i - dp[i-1] - 1
                if prev >= 0 && s[prev] == '(' {
                    dp[i] = dp[i-1] + 2
                    if prev > 0 {
                        dp[i] += dp[prev-1]
                    }
                }
            }
            if dp[i] > maxLen {
                maxLen = dp[i]
            }
        }
    }
    
    return maxLen
}

方法二:栈

go 复制代码
func longestValidParentheses(s string) int {
    maxLen := 0
    stack := []int{-1} // 初始化栈底元素
    
    for i, char := range s {
        if char == '(' {
            stack = append(stack, i)
        } else {
            stack = stack[:len(stack)-1] // 弹出栈顶元素
            if len(stack) == 0 {
                stack = append(stack, i) // 记录新的基准点
            } else {
                currentLen := i - stack[len(stack)-1]
                if currentLen > maxLen {
                    maxLen = currentLen
                }
            }
        }
    }
    
    return maxLen
}

方法三:双向遍历(最优空间复杂度)

go 复制代码
func longestValidParentheses(s string) int {
    left, right, maxLen := 0, 0, 0
    
    // 从左到右遍历
    for i := 0; i < len(s); i++ {
        if s[i] == '(' {
            left++
        } else {
            right++
        }
        
        if left == right {
            if 2*right > maxLen {
                maxLen = 2 * right
            }
        } else if right > left {
            left, right = 0, 0
        }
    }
    
    left, right = 0, 0
    // 从右到左遍历
    for i := len(s) - 1; i >= 0; i-- {
        if s[i] == '(' {
            left++
        } else {
            right++
        }
        
        if left == right {
            if 2*left > maxLen {
                maxLen = 2 * left
            }
        } else if left > right {
            left, right = 0, 0
        }
    }
    
    return maxLen
}

使用示例

go 复制代码
package main

import "fmt"

func main() {
    fmt.Println(longestValidParentheses("(()"))      // 输出: 2
    fmt.Println(longestValidParentheses(")()())"))   // 输出: 4
    fmt.Println(longestValidParentheses(""))         // 输出: 0
    fmt.Println(longestValidParentheses("()(()"))    // 输出: 2
    fmt.Println(longestValidParentheses("()(())"))   // 输出: 6
}

方法比较

  1. 动态规划

    • 时间复杂度:O(n)
    • 空间复杂度:O(n)
    • 优点:直观易懂,容易实现
    • 缺点:需要额外的数组空间
    • 时间复杂度:O(n)
    • 空间复杂度:O(n)
    • 优点:思路清晰,利用栈的特性
    • 缺点:同样需要额外空间
  2. 双向遍历

    • 时间复杂度:O(n)
    • 空间复杂度:O(1)
    • 优点:空间效率最高
    • 缺点:需要两次遍历

这些方法都能有效解决最长有效括号子串长度的问题,可以根据具体需求选择合适的方法。

相关推荐
allan bull3 分钟前
在节日中寻找平衡:圣诞的欢乐与传统节日的温情
人工智能·学习·算法·职场和发展·生活·求职招聘·节日
似水এ᭄往昔19 分钟前
【C++】--封装红⿊树实现mymap和myset
开发语言·数据结构·c++·算法·stl
咕噜企业分发小米21 分钟前
腾讯云向量数据库HNSW索引如何更新?
人工智能·算法·腾讯云
lcreek26 分钟前
LeetCode215. 数组中的第K个最大元素、LeetCode912. 排序数组
python·算法·leetcode
无限大639 分钟前
为什么"数据压缩"能减小文件大小?——从冗余数据到高效编码
后端
用户7294294322339 分钟前
kubernetes/k8s全栈技术讲解+企业级实战项目课程
后端
Einsail40 分钟前
天梯赛题解(3-6)
算法
杜子不疼.41 分钟前
【LeetCode 852 & 162_二分查找】山脉数组的峰顶索引 & 寻找峰值元素
算法·leetcode·职场和发展
用户7294294322341 分钟前
基于Dubbo的分布式系统架构+事务解决方案
后端
程序员鱼皮43 分钟前
什么是 RESTful API?凭什么能流行 20 多年?
前端·后端·程序员