zoukankan      html  css  js  c++  java
  • No.022:Generate Parentheses

    问题:

    Given n pairs of parentheses, write a function to generate all combinations of well-formed parentheses.

    For example, given n = 3, a solution set is:

    [

      "((()))",

      "(()())",

      "(())()",

      "()(())",

      "()()()"

    ]

    官方难度:

    Medium

    翻译:

    合并n个括号,生成所有n个括号组成的合理序列。

    例子:

    给定n=3,解集为

    [

      "((()))",

      "(()())",

      "(())()",

      "()(())",

      "()()()"

    ]

    方法一:

    1. 初始的思想,考虑使用递归,每次优先获取n-1个括号所有情况下的集合,然后插入新的括号。
    2. 递归的终点是n=1,返回包含一个元素“()”的集合。
    3. 每次先加“(”,在第一个位置和一个“)”的下一个位置,插入左括号。
    4. 然后加右括号,从加入的左括号开始,统计左括号和右括号的个数,在左括号个数大于右括号个数的情况下,才可以加入右括号。
    5. 不难发现,虽然已经极力避免了,但是这种方法会出现很多的重复情况,所以需要一个Set集合去重,在递归结束之后,转化成List集合返回。

    方法一的解题代码:

     1     // 根据n-1的情况增加括号
     2     public static List<String> method(int n) {
     3         List<String> list = new ArrayList<>(getPharentheses(n));
     4         return list;
     5     }
     6 
     7     private static Set<String> getPharentheses(int n) {
     8         Set<String> set = new HashSet<>();
     9         if (n == 1) {
    10             set.add("()");
    11         } else {
    12             Set<String> last = getPharentheses(n - 1);
    13             for (String s : last) {
    14                 String s1 = "(" + s;
    15                 int l1 = 0;
    16                 int r1 = 0;
    17                 for (int j = 1; j < s1.length(); j++) {
    18                     if (s1.charAt(j) == '(') {
    19                         l1++;
    20                     } else {
    21                         r1++;
    22                     }
    23                     // 这里有重复的可能性
    24                     if (l1 > r1) {
    25                         set.add(s1.substring(0, j + 1) + ")" + s1.substring(j + 1));
    26                     }
    27                 }
    28                 // 先加左括号
    29                 for (int i = 1; i < s.length(); i++) {
    30                     if (s.charAt(i) == ')') {
    31                         String str = s.substring(0, i + 1) + "(" + s.substring(i + 1);
    32                         // 左右括号计数
    33                         int l = 0;
    34                         int r = 0;
    35                         for (int j = i + 1; j < str.length(); j++) {
    36                             if (str.charAt(j) == '(') {
    37                                 l++;
    38                             } else {
    39                                 r++;
    40                             }
    41                             // 这里有重复的可能性
    42                             if (l > r) {
    43                                 set.add(str.substring(0, j + 1) + ")" + str.substring(j + 1));
    44                             }
    45                         }
    46                     }
    47                 }
    48             }
    49         }
    50         return set;
    51     }
    method

    方法二:

    1. 显而易见,如果通过某种算法,没有重复的情况产生,那么方法的执行效率一定会快很多。
    2. 假设n=3。先加上所有的左括号,“(((”,那么右括号的可能性只有一个,那就是“((()))”。
    3. 然后回退一步,少加一个左括号,加上一个右括号,之后再加上这个左括号,会形成以下形式“(()())”。
    4. 后加的左括号再回退一步“(())()”。依次类推,直到这个左括号不能再加为止,回退至第一步,在一开始回退2个左括号,得到以下2种形式“()(())”和“()()()”。
    5. 发现回退的左括号为n时,结束。
    6. 解题的代码形式很简单,但是逻辑比较难捋通顺。
    7. 注意入参检查。

     

    方法二的解题代码:

     1     public static List<String> generateParenthesis(int n) {
     2         if (n < 1) {
     3             throw new IllegalArgumentException("Input error");
     4         }
     5         List<String> list = new ArrayList<>();
     6         generate(n, n, "", list);
     7         return list;
     8     }
     9 
    10     // 先加所有左括号,遍历所有可能性,然后回退一个左括号,依次类推
    11     private static void generate(int left, int right, String current, List<String> list) {
    12         // 先加左括号
    13         if (left > 0) {
    14             generate(left - 1, right, current + "(", list);
    15         }
    16         if (right > 0 && right > left) {
    17             generate(left, right - 1, current + ")", list);
    18         }
    19         if (left == 0 && right == 0) {
    20             list.add(current);
    21         }
    22     }
    generateParenthesis

    相关链接:

    https://leetcode.com/problems/generate-parentheses/

    https://github.com/Gerrard-Feng/LeetCode/blob/master/LeetCode/src/com/gerrard/algorithm/medium/Q022.java

    PS:如有不正确或提高效率的方法,欢迎留言,谢谢!

  • 相关阅读:
    Servlet的几种跳转(转)
    Java String.split()用法小结(转)
    表单数据提交的方法
    gedit文本编辑器乱码解决办法
    J-Link烧写bootloader到mini2440的Nor Flash
    虚拟机安装Fedora10系统遇到异常
    linux系统忘记root密码怎么办?
    编译busybox时出错及解决方案
    source insight代码查看器如何自定义添加文件类型
    < Objective-C >文件操作-NSFileHandle
  • 原文地址:https://www.cnblogs.com/jing-an-feng-shao/p/6126041.html
Copyright © 2011-2022 走看看