想要精通算法和SQL的成长之路 - 恢复二叉搜索树和有序链表转换二叉搜索树

想要精通算法和SQL的成长之路 - 恢复二叉搜索树和有序链表转换二叉搜索树

  • 前言
  • [一. 恢复二叉搜索树](#一. 恢复二叉搜索树)
  • [二. 有序链表转换二叉搜索树](#二. 有序链表转换二叉搜索树)

前言

想要精通算法和SQL的成长之路 - 系列导航

一. 恢复二叉搜索树

原题链接

首先,一个正常地二叉搜索树在中序遍历下,遍历的元素一定是单调递增的。而题目中提醒道,正好有两个节点被互换了。那么以下面的递增序列为例:

  • 1,2,3,4,5,6,7\]。

  • 第一个问题点:6,4。这里我们需要找到第一个:下一个元素比当前元素小的,我们取当前元素。取前者。
  • 第二个问题点:5,3,在找到第一个问题点的前提下,再找:下一个元素比当前元素小的,我们取后者。
  • 最后两个问题节点值互换即可(不可以节点互换,一定是值互换)
java 复制代码
TreeNode firstNode, secondNode, preNode = new TreeNode(Integer.MIN_VALUE);

public void recoverTree(TreeNode root) {
    inOrderReadTree(root);
    int tmp = firstNode.val;
    firstNode.val = secondNode.val;
    secondNode.val = tmp;
}

public void inOrderReadTree(TreeNode root) {
    if (root == null) {
        return;
    }
    // 左中右的顺序遍历
    inOrderReadTree(root.left);
    // 找到第一个开始递减的元素
    if (firstNode == null && preNode.val > root.val) {
        firstNode = preNode;// 取前者
    }
    // 找到第一个节点的前提下,找到第二个开始递减的元素
    if (firstNode != null && preNode.val > root.val) {
        secondNode = root;// 取后者
    }
    preNode = root;
    inOrderReadTree(root.right);
}

二. 有序链表转换二叉搜索树

原题链接

首先一点,题目的入参是一个链表,对于链表而言,不好操作,我们可以先把他转为数组:

java 复制代码
// 构建成数组,入参是head
ArrayList<Integer> tmp = new ArrayList<>();
while (head != null) {
    tmp.add(head.val);
    head = head.next;
}
int[] arr = new int[tmp.size()];
for (int i = 0; i < arr.length; i++) {
    arr[i]= tmp.get(i);
}

其次,既然要高度平衡,那么这个题目而言,我们必定是针对转化后的数组,不断地以中间节点为根节点,向左右两侧构建树。

那么就是一个很简单的分治递归法:

java 复制代码
public TreeNode buildTree(int left, int right, int[] arr) {
    if (left > right) {
        return null;
    }
    // 中间节点下标
    int mid = (left + right) >> 1;
    TreeNode root = new TreeNode(arr[mid]);
    root.left = buildTree(left, mid - 1, arr);
    root.right = buildTree(mid + 1, right, arr);
    return root;
}

完整代码如下:

java 复制代码
public TreeNode sortedListToBST(ListNode head) {
    // 构建成数组
    ArrayList<Integer> tmp = new ArrayList<>();
    while (head != null) {
        tmp.add(head.val);
        head = head.next;
    }
    int[] arr = new int[tmp.size()];
    for (int i = 0; i < arr.length; i++) {
        arr[i] = tmp.get(i);
    }
    TreeNode treeNode = buildTree(0, arr.length - 1, arr);
    return treeNode;
}

public TreeNode buildTree(int left, int right, int[] arr) {
    if (left > right) {
        return null;
    }
    // 中间节点下标
    int mid = (left + right) >> 1;
    TreeNode root = new TreeNode(arr[mid]);
    root.left = buildTree(left, mid - 1, arr);
    root.right = buildTree(mid + 1, right, arr);
    return root;
}
相关推荐
八股文领域大手子24 分钟前
深入理解缓存淘汰策略:LRU 与 LFU 算法详解及 Java 实现
java·数据库·算法·缓存·mybatis·哈希算法
noravinsc42 分钟前
django admin 中更新表数据 之后再将数据返回管理界面
数据库·django·sqlite
春风不会绿大地1 小时前
IDEA编写flinkSQL(快速体验版本,--无需配置环境)
sql·flink
__lost1 小时前
C++ 解决一个简单的图论问题 —— 最小生成树(以 Prim 算法为例)
算法·图论·最小生成树·prim算法
wuqingshun3141592 小时前
蓝桥杯 11. 打印大X
数据结构·算法·职场和发展·蓝桥杯·深度优先
Bruce-li__2 小时前
DRF凭什么更高效?Django原生API与DRF框架开发对比解析
数据库·django·sqlite
Blossom.1183 小时前
量子网络:构建未来通信的超高速“高速公路”
网络·opencv·算法·安全·机器学习·密码学·量子计算
A林玖3 小时前
【机器学习】朴素贝叶斯
人工智能·算法·机器学习
六边形战士DONK3 小时前
神经网络基础[损失函数,bp算法,梯度下降算法 ]
人工智能·神经网络·算法
noravinsc3 小时前
connection.cursor() 与 models.objects.filter
数据库·django·原生查询·orm查询