括号生成
我的解答:
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可能性拼接"的解题方法。