「力扣」第 95 题:不同的二叉搜索树 II
给定一个整数 n,生成所有由 1 … n 为节点所组成的二叉搜索树。
示例:
输入: 3 输出: [ [1,null,3,2], [3,2,null,1], [3,1,null,null,2], [2,1,3], [1,null,2,null,3] ] 解释: 以上的输出对应以下 5 种不同结构的二叉搜索树: 1 3 3 2 1 \ / / / \ \ 3 2 1 1 3 2 / / \ \ 2 1 2 3
方法一:递归(分治法)
Java 代码:
import java.util.ArrayList;
import java.util.List;
class TreeNode {
int val;
TreeNode left;
TreeNode right;
TreeNode(int x) {
val = x;
}
}
public class Solution {
// 递归(分支)
public List<TreeNode> generateTrees(int n) {
if (n == 0) {
return new ArrayList<>();
}
return generateTrees(1, n);
}
private List<TreeNode> generateTrees(int left, int right) {
List<TreeNode> res = new ArrayList<>();
if (left > right) {
// 上层调用的方法须要这个空结点作为其左结点或者右节点
res.add(null);
return res;
}
if (left == right) {
// 只有一个结点,这个结点作为根结点返回即可
// 这一步可以包括到下面一个情况中
res.add(new TreeNode(left));
return res;
}
for (int i = left; i <= right; i++) {
List<TreeNode> leftList = generateTrees(left, i - 1);
List<TreeNode> rightList = generateTrees(i + 1, right);
for (TreeNode leftTree : leftList) {
for (TreeNode rightTree : rightList) {
TreeNode root = new TreeNode(i);
root.left = leftTree;
root.right = rightTree;
res.add(root);
}
}
}
return res;
}
}
可以把 left == right
的情况去掉。
Java 代码:
import java.util.ArrayList;
import java.util.List;
public class Solution2 {
public List<TreeNode> generateTrees(int n) {
if (n == 0) {
return new ArrayList<>();
}
return generateTrees(1, n);
}
private List<TreeNode> generateTrees(int left, int right) {
List<TreeNode> res = new ArrayList<>();
if (left > right) {
// 这个位置要占住
res.add(null);
return res;
}
for (int i = left; i <= right; i++) {
for (TreeNode leftTree : generateTrees(left, i - 1)) {
for (TreeNode rightTree : generateTrees(i + 1, right)) {
TreeNode root = new TreeNode(i);
root.left = leftTree;
root.right = rightTree;
res.add(root);
}
}
}
return res;
}
}
方法二:动态规划
Java 代码:
import java.util.ArrayList;
import java.util.List;
/**
* 动态规划的解法
*/
public class Solution4 {
public List<TreeNode> generateTrees(int n) {
List<TreeNode>[] res = new ArrayList[n + 1];
res[0] = new ArrayList<>();
if (n <= 0) {
return res[0];
}
// 注意这个位置
res[0].add(null);
for (int i = 1; i <= n; i++) {
// 先将对象数组初始化
res[i] = new ArrayList<>();
for (int j = 0; j < i; j++) {
for (TreeNode left : res[j]) {
for (TreeNode right : res[i - j - 1]) {
TreeNode root = new TreeNode(j + 1);
root.left = left;
root.right = clone(right, j + 1);
res[i].add(root);
}
}
}
}
return res[n];
}
private TreeNode clone(TreeNode root, int k) {
if (root == null) {
return root;
}
TreeNode curNode = new TreeNode(root.val + k);
curNode.left = clone(root.left, k);
curNode.right = clone(root.right, k);
return curNode;
}
}