2116. Check if a Parentheses String Can Be Valid - Detailed Explanation

Free Coding Questions Catalog
Boost your coding skills with our essential coding questions catalog. Take a step towards a better tech career now!

Problem Statement

You are given a string s containing only the characters '(', ')', and '*'. The character '*' can represent:

  • a left parenthesis '(',
  • a right parenthesis ')', or
  • an empty string (i.e. it can be removed).

Return true if s can be made valid after replacing every '*' with one of the above possibilities. A string is valid if:

  1. Every left parenthesis '(' has a corresponding right parenthesis ')'.
  2. Every right parenthesis ')' has a corresponding left parenthesis '('.
  3. Left parentheses must appear before the corresponding right parentheses.
  4. An empty string is valid.

Examples

Example 1

Input:

s = "()*"

Output:

true

Explanation:

  • The '*' can be treated as an empty string, making the string "()" which is valid.
  • Alternatively, '*' could represent either '(' or ')' but at least one replacement makes it valid.

Example 2

Input:

s = "(*)"

Output:

true

Explanation:

  • Here, '*' can be an empty string, resulting in "()" which is valid.

Example 3

Input:

s = "(*))"

Output:

true

Explanation:

  • The '*' can be interpreted as '(', which yields "(()))", and then the extra right parenthesis can be balanced by the chosen replacement. Alternatively, by different replacement choices, one can balance the string.

Hints

  1. Greedy Approach:

    • Think in terms of tracking the range of possible open parenthesis counts as you scan the string.
    • Use two counters: one for the lowest possible number of open parentheses and one for the highest.
    • Update these counters when you see '(', ')', or '*' (which can act as (, ), or empty).
  2. Dynamic Programming:

    • Consider DP where dp[i][j] means whether the first i characters can achieve exactly j open parentheses.
    • This approach is more straightforward but less efficient.
  3. Stack Approach:

    • Use stacks to keep track of indices for '(' and '*'. Then try to match any unmatched '(' with available '*'.
    • Finally, ensure that any remaining '(' can be balanced by an appropriate '*' coming later.

Approach 1: Greedy (Optimal) Solution

Idea

  • Traverse the string while maintaining two variables:
    • lo: the lowest possible number of open left parentheses.
    • hi: the highest possible number of open left parentheses.
  • For each character:
    • If it’s '(': increment both lo and hi.
    • If it’s ')': decrement both lo and hi.
    • If it’s '*': decrement lo (treat as ')') and increment hi (treat as '(').
  • Clamp lo to a minimum of 0 (you cannot have a negative number of open parentheses).
  • If at any point hi becomes negative, it means there are too many ')' and no replacement can fix that.
  • At the end, if lo is 0, the string can be valid.

Step-by-Step Walkthrough with Table

Consider s = "(*))":

StepCharacterActionlo updatehi updateComment
1(It’s '(' → add one open parenthesislo = 0 + 1 = 1hi = 0 + 1 = 1One open parenthesis
2*It’s '*' → can be ')' (decrease) or '(' (increase)lo = 1 - 1 = 0hi = 1 + 1 = 2Range becomes [0, 2]
3)It’s ')' → decrease open countlo = 0 - 1 = -1 → clamp to 0hi = 2 - 1 = 1lo clamped: [0, 1]
4)It’s ')' → decrease open countlo = 0 - 1 = -1 → clamp to 0hi = 1 - 1 = 0Final range: [0, 0] (valid since lo==0)

Since lo is 0 at the end, the string can be valid.

Complexity Analysis

  • Time Complexity: O(n), where n is the length of the string.
  • Space Complexity: O(1) (constant extra space for counters).

Python Code

Python3
Python3

. . . .

Java Code

Java
Java

. . . .

Approach 2: Dynamic Programming (Alternative)

Idea

Define dp[i][j] to be true if the first i characters can have exactly j open parentheses.

  • Start with dp[0][0] = true.
  • For each character:
    • If it’s '(': then for each j, dp[i+1][j+1] |= dp[i][j].
    • If it’s ')': then for each j > 0, dp[i+1][j-1] |= dp[i][j].
    • If it’s '*': it can be treated as '(', ')', or empty.
  • At the end, if any dp[n][0] is true, the string can be valid.

Complexity

  • Time Complexity: O(n²) where n is the length of the string.
  • Space Complexity: O(n²).

Python Code

Python3
Python3

. . . .

Java Code

Java
Java

. . . .

Common Mistakes & Edge Cases

  • Negative Open Count:
    Ensure that the lower bound of open parentheses never becomes negative. In the greedy approach, we use max(lo, 0) after each update.

  • Early Termination:
    If at any point the maximum possible count (hi) becomes negative, the string cannot be valid.

  • All '*' String:
    A string of all '*' is always valid because each '*' can be treated as an empty string.

  • Empty String:
    An empty string should return true.

  • Balance Between Choices:
    In the DP approach, ensure you consider all three possibilities for '*' (empty, (, )).

  • Valid Parentheses:
    Given a string containing just the characters '(', ')', '{', '}', '[', and ']', determine if the input string is valid. (This problem is simpler since no wildcard characters are involved.)

  • Longest Valid Parentheses:
    Find the length of the longest valid (well-formed) parentheses substring.

  • Remove Invalid Parentheses:
    Remove the minimum number of invalid parentheses to make the input string valid and return all possible results.

  • Generate Parentheses:
    Generate all combinations of well-formed parentheses for a given number of pairs.

Summary

We have covered two main approaches to solve the Check if a Parentheses String Can Be Valid problem:

  1. Greedy Approach:

    • Maintain a range of possible open parentheses counts using two counters (lo and hi).
    • Time Complexity: O(n), Space Complexity: O(1).
  2. Dynamic Programming Approach:

    • Use a DP table where dp[i][j] indicates if the first i characters can have exactly j open parentheses.
    • Time Complexity: O(n²), Space Complexity: O(n²).

Both approaches have their merits. The greedy solution is optimal in terms of time and space, while the DP solution is more intuitive for those who think in terms of state transitions.

TAGS
leetcode
CONTRIBUTOR
Design Gurus Team
-

GET YOUR FREE

Coding Questions Catalog

Design Gurus Newsletter - Latest from our Blog
Boost your coding skills with our essential coding questions catalog.
Take a step towards a better tech career now!
Explore Answers
What is SAR technique?
What tech stack does PayPal use?
How to describe yourself in mock interview?
Related Courses
Image
Grokking the Coding Interview: Patterns for Coding Questions
Grokking the Coding Interview Patterns in Java, Python, JS, C++, C#, and Go. The most comprehensive course with 476 Lessons.
Image
Grokking Modern AI Fundamentals
Master the fundamentals of AI today to lead the tech revolution of tomorrow.
Image
Grokking Data Structures & Algorithms for Coding Interviews
Unlock Coding Interview Success: Dive Deep into Data Structures and Algorithms.
Image
One-Stop Portal For Tech Interviews.
Copyright © 2025 Design Gurus, LLC. All rights reserved.
;