题目描述
给定一个只包括 ' ( ' , ' ) ', ' { ' , ' } ' , ' [ ' , ' ] ' 的字符串,判断字符串是否有效。
有效字符串需满足:
- 左括号必须用相同类型的右括号闭合。
- 左括号必须以正确的顺序闭合。
注意空字符串可被认为是有效字符串。
示例1:
输入:"()"
输出:true
示例2:
输入:"()[ ] { } "
输出:true
示例3:
输入:"( ] "
输出:false
示例4:
输入:"( [ )] "
输出:false
示例5:
输入:" { [ ] } "
输出:true
简化版本
让我们看一下该问题的简化版本,在简化后的问题中,只含一种类型的括号。这么一来,我们将会遇到的表达式是
( ( ( ( ( ( ) ) ) ) ) ) - VALID
( ) ( ) ( ) ( ) -VALID
( ( ( ( ( ( ( ( ) -INVALID
( ( ( ) ( ( ) ) ) ) - VALID
我们试着用一个简单的算法来解决这一问题。
1. 我们从表达式的左侧开始,每次只处理一个括号。
2. 假设我们遇到一个开括号即(,表达式是否无效取决于这个表达式的其它部分是否有相匹配的闭括号即 )。此时,我们只是增加计数器的值保持跟踪现在为止开括号的数目。left += 1 。
3. 如果我们遇到一个闭括号,这可能意味着这样两种情况:
· 此闭括号没有与之对应的开括号,在这种情况下,我们的表达式无效。当 left == 0,也就是没有未配对的左括号可用时就是这种情况。
· 我们有一些未配对的开括号可以与该闭括号配对。当1eft>0,也就是有未配对的左括号可用时就是这种情况。
4. 如果我们在 left == 0时遇到一个闭括号例如 ),那么当前的表达式无效。否则,我们会减少 left 的值,也就是减少了可用的未配对的左括号的数量。
5. 继续处理字符串,直到处理完所有括号。
6. 如果最后我们仍然有未配对的左括号,这意味着表达式无效。
在这里讨论这个特定算法是因为我们从该解决方案中获得灵感以解决原始问题。为了更好地理解我们讨论的算法,请观看下面的动画演示。
如果我们对原始问题这个办法,这是根本就行不通的。基于简单计数器的方法能够在上面完美运行是因为所有括号都具有相同的类型。
因此,当我们遇到一个闭括号时,我们只需要假设有一个对应匹配的开括号是可用的,即假设 left > 0。
但是,在我们的问题中,如果我们遇到 ]
,我们真的不知道是否有相应的 [ 可用。你可能会问:
为什么不为不同类型的括号分别维护一个单独的计数器?
这可能不起作用,因为括号的相对位置在这里也很重要。例如:
[ { ]
如果我们只是在这里维持计数器,那么只要我们遇到闭合方括号,我们就会知道此处有一个可用的未配对的开口方括号。
但是,最近的未配对的开括号是一个花括号,而不是一个方括号,因此计数方法在这里被打破了。
方法一:栈
关于有效括号表达式的一个有趣属性是有效表达式的子表达式也应该是有效表达式。(不是每个子表达式)例如