Skip to content

LeetCode 20 LintCode 423 Valid Parentheses (stack)

Updated: at 05:12 PM

Table of contents

Open Table of contents

Description

Given a string s containing just the characters '(', ')', '{', '}', '[' and ']', determine if the input string is valid.

An input string is valid if:

Open brackets must be closed by the same type of brackets. Open brackets must be closed in the correct order. Every close bracket has a corresponding open bracket of the same type.

Example 1:

Input: s = "()"

Output: true

Example 2:

Input: s = "()[]{}"

Output: true

Example 3:

Input: s = "(]"

Output: false

Example 4:

Input: s = "([])"

Output: true


Constraints:

1 <= s.length <= 10^4
s consists of parentheses only '()[]{}'.

Solution

Idea

We use a stack to memorize the open brackets seen and their order. When we see a close bracket, we pop the most recent bracket (LIFO) to match. Remember to check whether there are any remaining brackets left on the stack at the end.

Java

Using Java 13 enhanced case switch, a bit shorter than regular if else.

class Solution {
    public boolean isValid(String s) {
        Stack<Character> stack = new Stack<>();
        for (int i = 0; i < s.length(); i++) {
            switch (s.charAt(i)) {
                case '(' -> stack.push(')');
                case '[' -> stack.push(']');
                case '{' -> stack.push('}');
                default -> {
                    if (stack.isEmpty() || s.charAt(i) != stack.pop()) return false;
                }
            }
        }
        return stack.isEmpty();
    }
}

Rust

In many cases, I found Rust tends to be verbose. But for this question, Rust is perfect. For the fourth branch, no separate emptiness check is needed because Rust’s VecDeque returns an Option.

use std::collections::VecDeque;

impl Solution {
    pub fn is_valid(s: String) -> bool {
        let mut st = VecDeque::new();
        for c in s.chars() {
            match c {
                '(' => st.push_back(')'),
                '[' => st.push_back(']'),
                '{' => st.push_back('}'),
                _ => if st.pop_back() != Some(c) {return false;}
            }
        }
        st.is_empty();
    }
}

Python

We can leverage match case introduced in Python 3.10.

class Solution:
    """0 ms, 17.42 mb"""

    def isValid(self, s: str) -> bool:
        st = deque()
        for c in s:
            match c:  # match case since 3.10
                case '(':
                    st.append(')')
                case '[':
                    st.append(']')
                case '{':
                    st.append('}')
                case _:
                    if not st or st.pop() != c:
                        return False
        return len(st) == 0

C++

C++ does not support returning the element with pop. So we needed one more branch in the if else branches.

class Solution {
public:
    bool isValid(string s) {
        deque<char> st;
        for (char c: s) {
            if (c == '(') st.push_back(')');
            else if (c == '[') st.push_back(']');
            else if (c == '{') st.push_back('}');
            else if (!st.empty() && st.back() == c) st.pop_back();
            else return false;
        }
        return st.empty();
    }
};

Previous Post
LeetCode 3097 Shortest Subarray With OR at Least K II (bit)
Next Post
Database (MongoDB, Microsoft SQL Server) CheatSheet