括号生成问题

该博客探讨了如何判断一个括号序列是否合法,以及如何生成所有可能的合法括号序列。提供了两种方法:一种是生成所有序列然后进行合法性检查,另一种是在生成过程中剔除不合法序列。每种方法都通过遍历和平衡计数来实现。
摘要由CSDN通过智能技术生成

题目

在这里插入图片描述

分析

首先明确一个问题:如何判断一个括号序列是不是合法的序列?
方法就是遍历这个序列,同时初始化一个x = 0,如果遇到左括号x就加一,遇到右括号就减一,如果遍历过程中x小于0,那么这个序列一定不是合法的序列;遍历完成后如果x等于0就是合法序列,否则是不合法的序列。代码如下:

//检验序列是否是合法的括号序列,current是序列值
 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;
    }

第一种方式:生成所有的序列,判断其中合法的序列

class Solution {
    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;
    }
}

第二种方式:对第一种方式进行了优化,我们知道左括号的最大数量就是n,而右括号的添加必须小于等于左括号的数量,因此可以对这些不合法的序列进行剔除。

public class Solution {
    public ArrayList<String> generateParenthesis (int n) {
        // write code here
        ArrayList<String> res = new ArrayList();
        StringBuffer buffer = new StringBuffer();
        generate(buffer,0,0,n,res);
        return res;
    }
    /*参数说明:
    1. buffer:保存中间生成的括号序列
    2. left:左括号的数量
    3. right:右括号的数量
    4. max;合法序列左右括号的最大数量
    5. res:保存所有的合法括号序列
    */
    public void generate(StringBuffer buffer,int left,int right,int max,ArrayList<String> res) {
        if(buffer.length() == max * 2) {
            res.add(buffer.toString());
            return;
        }
        if(left < max) {//如果左括号的数量小于最大数量
            buffer.append('(');
            generate(buffer,left + 1,right,max,res);
            buffer.deleteCharAt(buffer.length() - 1);
        }
        if(right < left) {//如果右括号的数量小于左括号的数量
             buffer.append(')');
            generate(buffer,left,right + 1,max,res);
            buffer.deleteCharAt(buffer.length() - 1);
        }
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值