22. 括号生成

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

dfs回溯入门题。
忘了回溯是啥感觉了,赶紧找个入门题做一下体会体会。先贴一段话体会一下:

回溯算法实际上一个类似枚举的搜索尝试过程,主要是在搜索尝试过程中寻找问题的解,当发现已不满足求解条件时,就 “回溯” 返回,尝试别的路径。回溯法是一种选优搜索法,按选优条件向前搜索,以达到目标。但当探索到某一步时,发现原先选择并不优或达不到目标,就退回一步重新选择,这种走不通就退回再走的技术为回溯法,而满足回溯条件的某个状态的点称为 “回溯点”。许多复杂的,规模较大的问题都可以使用回溯法,有“通用解题方法”的美称。
回溯算法的基本思想是:从一条路往前走,能进则进,不能进则退回来,换一条路再试。

然后草稿纸上画一棵树,就能知道条件是啥了,开始写代码。
注意点:

  1. 满足退出条件 (✅),不满足退出的条件(忘写了,导致死循环)。
  2. vector results在递归中是个全局的角色,所以要用引用,这样递归函数对它的修改才可以保存下来传递下去。不然就只是拷贝,没啥用。
class Solution {
public:
    vector generateParenthesis(int n) {
        vector results;
        dfs(results, "", 0, 0, n);
        return results;
    }

    void dfs(vector& results, string str, int left, int right, int n) {
        if (left == n && right == n) {
            results.push_back(str);
            return;
        }
        if (left > n || right > n) return; // 忘写了,导致死循环
        if (left > right) {
            dfs(results, str+'(', left+1, right, n);
            dfs(results, str+')', left, right+1, n);
        } else {
            dfs(results, str+'(', left+1, right, n);
        }
    }
};

更简洁的写法参考一下,能写这么简洁,主要归功于条件想得十分清楚。

class Solution {
public:
    vector generateParenthesis(int n) {
        vector res;
        dfs(n, 1, 0, "(", res);
        return res;
    }

    void dfs(int n, int left, int right, string str, vector& res) {
        if (left == n && right == n) { 
            res.push_back(str);
            return;
        }
        if (left < right || left > n || right > n) return;
        dfs(n, left + 1, right, str+'(', res);
        dfs(n, left, right + 1, str+')', res);
    }
};

你可能感兴趣的:(22. 括号生成)