题解 | #合法的括号字符串#

合法的括号字符串

http://www.nowcoder.com/practice/eceb50e041ec40bd93240b8b3b62d221

使用两个栈,一个存放左括号,一个存放*,如果碰到右括号,先和左括号栈里的进行抵消,如果左括号栈为空再和*栈的抵消。最后再判断左括号栈长度是否小于等于*栈,满足的话说明剩余的左括号也都可以抵消,但是这样忽略了左括号和*的有效作用区间,也就是入栈时间,比如**((),最后虽然*栈长度大于左括号栈,但是那些*是在左括号之前的,无法与左括号进行抵消。

因此还需要两个栈来分别计算左括号栈和*栈中数据的入栈时间(可以用在数组中的index),它们与左括号栈和*栈同时进行push(),pop()操作。

最后遍历两个时间栈,如果存在左括号的时间>*的时间,说明不合法返回false。

import java.util.*;


public class Solution {
    /**
     * 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可
     *
     * 
     * @param s string字符串 
     * @return bool布尔型
     */
    public boolean isValidString (String s) {
        // write code here
        char[] str = s.toCharArray();
        // 存放左括号
        Deque<Character> stack = new LinkedList<>();
        // 记录左括号栈里的左括号入栈时间
        Deque<Integer> stackNum = new LinkedList<>();
        // 存放*号
        Deque<Character> stack2 = new LinkedList<>();
        // 记录*栈中 *的入栈时间
        Deque<Integer> stackNum2 = new LinkedList<>();
        for(int i = 0;i<str.length;i++){
            if(str[i] == '('){
                stack.push(str[i]);
                stackNum.push(i);
            }else if(str[i] == '*'){
                stack2.push(str[i]);
                stackNum2.push(i);
            }else{ // 如果遇到右括号
                // 先判断 stack是否为空
                if(!stack.isEmpty()){
                    stack.pop();
                    stackNum.pop();
                }else{
                    if(stack2.isEmpty()){
                        return false;
                    }else{
                        stack2.pop();
                        stackNum2.pop();
                    }
                }
            }
        }
        
        while(!stack.isEmpty() && !stack2.isEmpty()){
            // 如果左括号入栈时间比*晚 返回false
            // 因为即使有* 但是它无法作用到后面的左括号 比如 **( 就是不合法的
            if(stackNum.pop() > stackNum2.pop())
                return false;
            stack.pop();
            stack2.pop();
        }
        return stack.isEmpty();
    }
}
全部评论

相关推荐

2025-12-28 20:47
已编辑
北京工商大学 Java
程序员牛肉:我靠你这个实习经历其实最需要担心的点是你做的太多了,可能会被面试官怀疑是你伪造的。 交易状态机是你做的,支付多渠道是你做的,对账是你做的,结算还是你做的,重复支付也是你做的,整个服务的异常处理也是你做的。 其实你这个反而问题很大的,你想想站在面试官的角度,他是真的会相信你的能力很强,还是相信这份实习你伪造了大部分?我相信你真的做了这么多,但是删一些,废话删一删。你这个做的太多了反而真实性不可信。 后面再补一个项目,在github上找一个高star的项目学一学然后写到自己简历上。我觉得你能力肯定没问题。28届能做到这个份上很厉害,但是在求职市场中,你不是在跟28届的同学比,把你这个简历放到27届其实也就一般水平。 所以后续要想一想看看能不能给自己简历上搞点亮点,比如开源贡献呢?比如博客呢?
实习要如何选择和准备?
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务