LeetCode 32. 最长有效括号

最长有效括号

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

方法一、栈

由于要存在配对,必然有')'存在,配对的长度为上一个')'到当前')'之间的距离,即为最长配对子串,因此,我们使用栈结构,存储上一次配对的')'所在的索引。栈存放**[最后一个没有被匹配的右括号的下标]**

复杂度分析

时间复杂度:O(n),n为字符串的长度,遍历一次

空间复杂度:O(n),n为字符串的长度,栈最多存储n个字符

Swift

swift 复制代码
func longestValidParentheses(_ s: String) -> Int {
        let cnt = s.count
        
        var res:Int = 0
        
        //栈,存放[最后一个没有被匹配的右括号的下标]
        let stack = Stack<Int>()
        stack.push(-1)//为了统一期间,假数据填充

        for i in 0..<cnt {
            let chara = s[s.index(s.startIndex, offsetBy: i)]
            if chara == "(" {
                stack.push(i)
            }else if(chara == ")") {
                let _ = stack.pop()
                
                if stack.count() == 0 {
                    stack.push(i)
                }else {
                    res = max(res, i - (stack.top() ?? 0))
                }
            }
        }
        return res
    }

OC

c 复制代码
- (NSInteger)longestValidParentheses:(NSString *)s {
    NSInteger res = 0;
    
    //存放[最后一个没有被匹配的右括号的下标]
    StackOC *stack = [[StackOC alloc] init];
    [stack push:@(-1)];//
    
    for (NSInteger i=0; i<s.length; i++) {
        unichar character = [s characterAtIndex:i];
        if (character == '(') {
            [stack push:@(i)];
        }else if(character == ')') {
            [stack pop];
            
            if ([stack empty]) {
                [stack push:@(i)];
            }else {
                //当前右括号索引减去上一个右括号的索引,表示最长串
                res = MAX(res, i-[[stack top] integerValue]);
            }
        }
    }
    
    return res;
}

方法二、双指针+双向遍历

思想:left对应 '('个数,right对应')'个数,当两者相等时,说明配对成功。

边界处理:当从左向右遍历时,right>left表示不连续了,要从0开始计数,因此将left和right置为0。

当从右向左遍历时,当left>right时表示不连续了,计数归0

复杂度分析

时间复杂度:O(n),n为字符串的长度,遍历一次

空间复杂度:O(1)

Swift

swift 复制代码
//双指针法,空间复杂度为O(1)
    func longestValidParentheses(_ s: String) -> Int {
        let cnt = s.count
        
        var left = 0, right = 0, maxLen = 0
        
        //从左向右找一次
        for i in 0..<cnt {
            let str = s[s.index(s.startIndex, offsetBy: i)]
            if str == "(" {
                left += 1
            }else if(str == ")") {
                right += 1
            }
            
            if left == right {
                maxLen = max(maxLen, 2*right)
            }else if(right > left) {
                left = 0
                right = 0
            }
        }
        
        //从右向左找一次
        left = 0; right = 0
        var j = cnt - 1
        while j > 0 {
            let str = s[s.index(s.startIndex, offsetBy: j)]
            
            if str == "(" {
                left += 1
            }else if(str == ")") {
                right += 1
            }
            
            if left == right {
                maxLen = max(maxLen, 2*right)
            }else if(left > right) {
                left = 0
                right = 0
            }
            
            j -= 1
        }
        
        return maxLen
    }

OC

c 复制代码
//双指针+左右遍历
- (NSInteger)longestValidParentheses:(NSString *)s {
    NSInteger left = 0, right = 0, maxLen = 0;
    
    //从左向右匹配一次
    for (NSInteger i=0; i<s.length; i++) {
        unichar charater = [s characterAtIndex:i];
        if (charater == '(') {
            left++;
        }else if(charater == ')') {
            right++;
        }
        
        if (left == right) {
            maxLen = MAX(maxLen, 2 * right);
        }else if(right > left) {
            left = right = 0;
        }
    }
    
    //从右向左匹配一次
    for (NSInteger i=s.length-1; i>0; i--) {
        unichar charater = [s characterAtIndex:i];
        if (charater == '(') {
            left++;
        }else if(charater == ')') {
            right++;
        }
        
        if (left == right) {
            maxLen = MAX(maxLen, 2*left);
        }else if(left > right) {
            left = right = 0;
        }
    }
    
    return maxLen;
}
相关推荐
艾莉丝努力练剑13 小时前
【Linux】初始Linux:从计算机历史发展、操作系统历史脉络的角度详谈Linux相关的话题,附Linux安装和用户创建(环境准备)详解
linux·运维·服务器·经验分享
l1t14 小时前
how to build tbox xml into the demo
xml·linux·c语言·parser·tbox
努力学习的小廉14 小时前
深入了解linux系统—— 线程池
linux·运维·服务器
不是编程家14 小时前
Linux第十五讲:Socket编程UDP
linux·运维·udp
lifallen14 小时前
揭秘KafkaStreams 线程缓存:NamedCache深度解析
数据结构·算法·缓存·kafka·apache
我的知识太少了14 小时前
P1122 最大子树和
算法
UrSpecial14 小时前
Linux线程
linux·开发语言·c++
郝学胜-神的一滴14 小时前
深入浅出 C++20:新特性与实践
开发语言·c++·程序人生·算法·c++20
格林威14 小时前
Linux使用-MySQL的使用
linux·运维·人工智能·数码相机·mysql·计算机视觉·视觉检测
程序员TNT14 小时前
Shoptnt 促销计算引擎详解:策略模式与责任链的完美融合
linux·windows·策略模式