LeetCode-有效的括号

LeetCode-有效的括号

LeetCode-有效的括号

1 Easy-有效的括号

1.1 题目描述

给定一个只包括 '(' ,')' , '{' , '}' , '[' , ']' 的字符串,判断字符串是否有效。

有效字符串需满足:

左括号必须用相同类型的右括号闭合。

左括号必须以正确的顺序闭合。

注意空字符串可被认为是有效字符串。

1.2 示例 1:

输入: "()"

输出: true

1.3 示例 2:

输入: "()[]{}"

输出: true

1.4 示例 3:

输入: "(]"

输出: false

1.5 示例 4:

输入: "([)]"

输出: false

1.6 示例 5:

输入: "{[]}"

输出: true

2 自己的答案

2.1 思路

  1. 首先排除不需要循环即可判断不是符合的字符串,例如
    • 空字符串
    • 字符串长度小于2(不能组成一对有效的括号)
    • 第一个字符是右括号的字符串
  2. 使用循环判断是否为有效字符串时,使用栈记录左括号的顺序,使用索引进行表示,并且索引是对称的(left和right变量,left[0]和right[0]为一对有效的括号,以此类推)
    • 当括号是左括号的时候,将左括号在left中对应的索引压入栈中
    • 当括号是右括号的时候,弹出栈vs中的一个值,该值为字符索引,通过该索引值与vs.pop(),栈弹出的值进行比较,如果不相等,不是有效的括号
  3. 注意的两个地方
    • 右括号的数量可能大于左括号的数量,所以在栈弹出值前要判断栈是否为空,空则直接返回为false,无效的括号
    • 如果左括号的值大于右括号的值,循环出来时候,需要判断栈是否为空,为空说明左括号数量和右括号数量相等,不为空是左括号数量大于右括号数量,返回false,,无效字符串

2.2 代码

package algorithm.easy;

import java.util.Stack;

public class ValidParentheses {
    public static boolean isVaild(String s) {
        String left = "([{";
        String right = ")]}";
        Stack<Integer> vs = new Stack<Integer>();
        boolean result = false;
        int leftPos = 0;
        int rightPos = 0;
        // 空字符串,返回true
        if (s.equals("")) {
            return true;
        }
        // 第一个字符串为空或字符串长度小于2或第一个字符是右括号,返回false
        if (s == null || s.length() < 2 || (right.indexOf(s.charAt(0)) != -1))
            return false;
        for (int i = 0; i < s.length(); i++) {
            // 如果该字符包含在左括号列表中,push到栈中
            leftPos = left.indexOf(s.charAt(i));
            if (leftPos != -1) {
                vs.push(leftPos);
            }
            // 如果该字符包含在右括号列表中,从栈中pop出来,并用来对比左括号与右括号的索引值是否相同
            rightPos = right.indexOf(s.charAt(i));
            if (rightPos != -1) {
                // 如果此时还是右括号,栈为空的话,说明右括号数量多于左括号数量,返回false
                if (vs.empty() || rightPos != vs.pop()) {
                    return false;
                }
            }
        }
        // 如果栈为空,返回true,否则返回false
        if (vs.empty()) {
            // 遍历均成功后,返回为有效字符串
            return true;
        } else {
            return false;
        }
    }
    public static void main(String[] args) {
        System.out.println(isVaild("()"));
        System.out.println(isVaild("()[]{}"));
        System.out.println(isVaild("(]"));
        System.out.println(isVaild("([)]"));
        System.out.println(isVaild("{[]}"));
    }
}

3 官方解答

3.1 思路

想象一下,你正在为你的大学课设编写一个小型编译器,编译器的任务之一(或称子任务)将检测括号是否匹配。

我们本文中看到的算法可用于处理编译器正在编译的程序中的所有括号,并检查是否所有括号都已配对。这将检查给定的括号字符串是否有效,是一个重要的编程问题。

我们这个问题中将要处理的表达式可以包含以下三种不同类型的括号:

(), {} 以及 [] 在查看如何检查由这些括号组成的给定表达式是否有效之前,让我们看一下该问题的简化版本,在简化后的问题中,只含一种类型的括号。这么一来,我们将会遇到的表达式是

(((((()))))) – VALID

()()()() – VALID

(((((((() – INVALID

((()(()))) – VALID 上我们试着用一个简单的算法来解决这一问题。

  1. 我们从表达式的左侧开始,每次只处理一个括号。
  2. 假设我们遇到一个开括号(即 ( ),表达式是否无效取决于在该表达式的其余部分的某处是否有相匹配的闭括号(即 ) )。此时,我们只是增加计数器的值保持跟踪现在为止开括号的数目。 left += 1
  3. 如果我们遇到一个闭括号,这可能意味着这样两种情况:
    • 此闭括号没有与与之对应的开括号,在这种情况下,我们的表达式无效。当 left == 0,也就是没有未配对的左括号可用时就是这种情况。
    • 我们有一些未配对的开括号可以与该闭括号配对。当 left > 0,也就是有未配对的左括号可用时就是这种情况。
  4. 如果我们在 left == 0 时遇到一个闭括号(例如 ) ),那么当前的表达式无效。否则,我们会减少 left 的值,也就是减少了可用的未配对的左括号的数量。
  5. 继续处理字符串,直到处理完所有括号。
  6. 如果最后我们仍然有未配对的左括号,这意味着表达式无效。

如果我们只是尝试对原始问题采用相同的办法,这是根本就行不通的。基于简单计数器的方法能够在上面完美运行是因为所有括号都具有相同的类型。因此,当我们遇到一个闭括号时,我们只需要假设有一个对应匹配的开括号是可用的,即假设 left > 0

但是,在我们的问题中,如果我们遇到 ] ,我们真的不知道是否有相应的 [ 可用。你可能会问:

为什么不为不同类型的括号分别维护一个单独的计数器?

这可能不起作用,因为括号的相对位置在这里也很重要。例如:

[{] 如果我们只是在这里维持计数器,那么只要我们遇到闭合方括号,我们就会知道此处有一个可用的未配对的开口方括号。但是,最近的未配对的开括号是一个花括号,而不是一个方括号,因此计数方法在这里被打破了。

3.2 方法

关于有效括号表达式的一个有趣属性是有效表达式的子表达式也应该是有效表达式。(不是每个子表达式)

此外,如果仔细查看上述结构,颜色标识的单元格将标记开闭的括号对。整个表达式是有效的,而它的子表达式本身也是有效的。这为问题提供了一种递归结构。例如,考虑上图中两个绿色括号内的表达式。开括号位于索引 1 ,相应闭括号位于索引 6

如果每当我们在表达式中遇到一对匹配的括号时,我们只是从表达式中删除它,会发生什么?

让我们看看下面的这个想法,从整体表达式中一次删除一个较小的表达式,因为这是一个有效的表达式,我们最后剩留下一个空字符串。

在表示问题的递归结构时,栈数据结构可以派上用场。我们无法真正地从内到外处理这个问题,因为我们对整体结构一无所知。但是,栈可以帮助我们递归地处理这种情况,即从外部到内部。

3.3 算法

初始化栈 S。

一次处理表达式的每个括号。

如果遇到开括号,我们只需将其推到栈上即可。这意味着我们将稍后处理它,让我们简单地转到前面的 子表达式。

如果我们遇到一个闭括号,那么我们检查栈顶的元素。如果栈顶的元素是一个 相同类型的 左括号,那么我们将它从栈中弹出并继续处理。否则,这意味着表达式无效。

如果到最后我们剩下的栈中仍然有元素,那么这意味着表达式无效。

实现的代码如下

class Solution {

  // Hash table that takes care of the mappings.
  private HashMap<Character, Character> mappings;

  // Initialize hash map with mappings. This simply makes the code easier to read.
  public Solution() {
    this.mappings = new HashMap<Character, Character>();
    this.mappings.put(')', '(');
    this.mappings.put('}', '{');
    this.mappings.put(']', '[');
  }

  public boolean isValid(String s) {

    // Initialize a stack to be used in the algorithm.
    Stack<Character> stack = new Stack<Character>();

    for (int i = 0; i < s.length(); i++) {
      char c = s.charAt(i);

      // If the current character is a closing bracket.
      if (this.mappings.containsKey(c)) {

        // Get the top element of the stack. If the stack is empty, set a dummy value of '#'
        char topElement = stack.empty() ? '#' : stack.pop();

        // If the mapping for this bracket doesn't match the stack's top element, return false.
        if (topElement != this.mappings.get(c)) {
          return false;
        }
      } else {
        // If it was an opening bracket, push to the stack.
        stack.push(c);
      }
    }

    // If the stack still contains elements, then it is an invalid expression.
    return stack.isEmpty();
  }
}

3.4 复杂度分析

  • 时间复杂度: \(O(n)\) ,因为我们一次只遍历给定的字符串中的一个字符并在栈上进行 \(O(1)\) 的推入和弹出操作。
  • 空间复杂度:\(O(n)\) ,当我们将所有的开括号都推到栈上时以及在最糟糕的情况下,我们最终要把所有括号推到栈上。例如 ((((((((((

Date: 2018-10-28 23:07

Author: devinkin

Created: 2018-10-29 一 13:05

Validate

posted @ 2018-10-28 23:10  EmacsDevinkin  阅读(186)  评论(0编辑  收藏  举报