059括号生成

括号生成

题目链接:https://leetcode.cn/problems/generate-parentheses/description/?envType=study-plan-v2\&envId=top-100-liked

我的解答:

复制代码
public List<String> generateParenthesis(int n) {
    List<String> ans = new ArrayList<>();
    backtreack(n, 0, 0, new StringBuffer(), ans);
    return ans;
}
//left:左括号的个数    right:右括号的个数
public void backtreack(int n, int left, int right, StringBuffer output, List<String> ans){
    if(right == n){
        ans.add(output.toString());
        return;
    }
    if(left + 1 <= n){
        //选取左括号
        left++;
        output.append('(');
        backtreack(n, left, right, output, ans);
        //回溯
        left--;
        output.deleteCharAt(output.length()-1);
    }

    if(right + 1 <= left){
        //选取右括号
        right++;
        output.append(')');
        backtreack(n, left, right, output, ans);
        //回溯
        right--;
        output.deleteCharAt(output.length()-1);
    }
}

分析:代码的时间复杂度计算复杂,详情请参考官方解析方法二中的时间复杂度分析,空间复杂度为O(n)。解题思路:采用递归 + 回溯方法,思路简单,故不赘述。

看了官方题解后的解答:

复制代码
//方法一:暴力法(思路简单,效率低,故直接粘贴了官方的解答)
//时间复杂度:O(n*2^2n),对于2^2n个序列中的每一个,我们用于建立和验证该序列的复杂度为 O(n)。
//空间复杂度:O(n)
public List<String> generateParenthesis(int n) {
    List<String> combinations = new ArrayList<String>();
    generateAll(new char[2 * n], 0, combinations);
    return combinations;
}

public void generateAll(char[] current, int pos, List<String> result) {
    if (pos == current.length) {
        if (valid(current)) {
            result.add(new String(current));
        }
    } else {
        current[pos] = '(';
        generateAll(current, pos + 1, result);
        current[pos] = ')';
        generateAll(current, pos + 1, result);
    }
}

public boolean valid(char[] current) {
    int balance = 0;
    for (char c: current) {
        if (c == '(') {
            ++balance;
        } else {
            --balance;
        }
        if (balance < 0) {
            return false;
        }
    }
    return balance == 0;
}

//方法二:回溯法(此方法与我的解答一致,但我参考官方解答将我的解答略微优化了一些多于的步骤)
//时间复杂度:本方法的时间复杂度计算复杂,详情请参考官方解析
//空间复杂度:O(n)
public List<String> generateParenthesis(int n) {
    List<String> ans = new ArrayList<>();
    backtreack(n, 0, 0, new StringBuffer(), ans);
    return ans;
}
//left:左括号的个数    right:右括号的个数
public void backtreack(int n, int left, int right, StringBuffer output, List<String> ans){
    if(right == n){
        ans.add(output.toString());
        return;
    }
    if(left < n){
        //选取左括号
        output.append('(');
        backtreack(n, left+1, right, output, ans);
        //回溯
        output.deleteCharAt(output.length() - 1);
    }

    if(right < left){
        //选取右括号
        output.append(')');
        backtreack(n, left, right+1, output, ans);
        //回溯
        output.deleteCharAt(output.length() - 1);
    }
}

//方法三:按括号序列的长度递归
//本方法的时间复杂度和空间复杂度计算复杂,详情请参考官方解析
ArrayList[] cache = new ArrayList[9];
public List<String> generateParenthesis(int n) {
    return generate(n);
}
public List<String> generate(int n){
    if(cache[n] != null){
        return cache[n];
    }
    ArrayList<String> res = new ArrayList<>();
    if(n == 0){
        res.add("");
    }
    else{
        for(int i=0; i<n; i++){
            for(String left : generate(i)){
                for(String right : generate(n-1-i)){
                    res.add("(" + left + ")" + right);
                }
            }
        }
    }
    cache[n] = res;
    return res;
}

分析:

​ 1、方法一采用暴力枚举,每次对枚举出的结果进行验证,若是有效答案,则加入结果。

​ 2、方法二在方法一的基础上进行了优化,递归前先通过左括号和右括号的个数进行判断,保证递归的答案一定是有效的。

​ 3、方法三的解题思路:对于每一个有效的括号序列,一定是以"("开头,以")"结尾的,所以每一个有效的括号序列都是 (a)b 的形式,其中a和b既可以为空,也可以是有效的括号序列。经过以上分析,我们只需递归计算 a 的所有可能和 b 的所有可能,遍历 a 与 b 的所有可能性并拼接,即可得到所有长度为 2n 的括号序列。另外,为了节省计算时间,我们可以在每次 generate(i) 函数返回之前,把返回值存储起来,下次再调用 generate(i) 时可以直接返回,不需要再递归计算。

总结

  • 本题只需掌握基本的递归与回溯即可。
  • 对于本题的方法三,我们需要善于观察与总结,关键在于"对于每一个有效的括号序列,一定是以"("开头,以")"结尾的,所以每一个有效的括号序列都是 (a)b 的形式,其中a和b既可以为空,也可以是有效的括号序列"这个结论,在这个结论的基础上可以很轻松的得出"a、b可能性拼接"的解题方法。
相关推荐
kisshyshy4 小时前
🍦 雪糕、食堂、火车厢:三幅漫画吃透栈、队列与链表
javascript·算法
猿人谷11 小时前
不只是 CPU 阈值:STAR 如何用 GAT + Transformer 做容器级自动扩缩容?
人工智能·算法
复杂网络13 小时前
Stable Diffusion 视觉大模型微调技术深度调研
算法
复杂网络13 小时前
基于 Stable Diffusion 架构的视觉大模型代表性工作与原理深度解析
算法
MrZhao40013 小时前
Agent Loop 如何用 Hook 扩展:权限、日志与工具拦截
算法
MrZhao40013 小时前
Agent 为什么需要 Skills:别把所有知识都塞进 system prompt
算法
JieE2122 天前
LeetCode 101. 对称二叉树|JS 递归 + 迭代双解法,彻底搞懂镜像判断
javascript·算法
JieE2123 天前
LeetCode 56. 合并区间|超清晰 JS 图解思路,面试高频区间题
javascript·算法·面试
Jack203 天前
HarmonyOS开发中错误处理策略:网络异常统一处理
算法
小小杨树3 天前
读懂色彩:拍照调色不再难
算法·计算机视觉·配色