题解 | #将升序数组转化为平衡二叉搜索树#

将升序数组转化为平衡二叉搜索树

http://www.nowcoder.com/practice/7e5b00f94b254da599a9472fe5ab283d

二叉搜索树(Binary Sort Tree)
二叉搜索树,又称之为二叉排序树(二叉查找树),它或许是一棵空树,或许是具有以下性质的二叉树:
图片说明
若他的左子树不为空,则左子树上所有节点的值都小于根节点的值
若它的右子树不为空,则右子树上所有节点的值都大于根节点的值
它的左右子树也分别是二叉搜索树

二叉平衡搜索树(AVL)
前面提到了二叉搜索树,我们知道,二叉搜索树的特性便于我们进行查找插入删除等一系列操作,其时间复杂度为O(logn),但是,如果遇见最差的情况,比如以下这棵树:
图片说明
这棵树,说是树,其实它已经退化成链表了,但从概念上来看,它仍是一棵二叉搜索树,只要我们按照逐次增大,如1、2、3、4、5、6的顺序构造一棵二叉搜索树,则形如上图。那么插入的时间复杂度就变成了O(n),导致这种糟糕的情况原因是因为这棵树极其不平衡,右树的重量远大于左树,因此我们提出了叫平衡二叉搜索树的结构,又称之为AVL树,是因为平衡二叉搜索树的发明者为Adel’son-Vel’skii 和Landis二人。
平衡二叉搜索树,它能保持二叉树的高度平衡,尽量降低二叉树的高度,减少树的平均查找长度。
AVL树的性质:
本质:仍是一棵二叉搜索树,只不过增加了平衡的要求

  1. 若他的左子树不为空,则左子树上所有节点的值都小于根节点的值
  2. 若它的右子树不为空,则右子树上所有节点的值都大于根节点的值
  3. 左子树与右子树高度之差的绝对值不超过1
  4. 树的每个左子树和右子树都是AVL树
  5. 每一个节点都有一个平衡因子(balance factor),任一节点的平衡因子是-1、0、1(每一个节点的平衡因子 = 右子树高度 - 左子树高度)

将升序数组转化为平衡二叉搜索树
题解一:二分+递归
题解思路: 利用BST中序遍历为升序序列的特性。
BST中序遍历:
[1,2,3,4,5,6,7]转换为AVL二叉搜索树后为:
图片说明
思路分析: 为了保持平衡,每次选择区间的中间值作为根节点。
递归分析:
递归边界: (left > right) 返回NULL;
递归过程: 选取中间值作为根节点,root = mid = (left+right+1)/2; 递归构建左右子树。

复杂度分析:
时间复杂度:O(N) 节点数N
空间复杂度:O(logN)

import java.util.*;

/*
 * public class TreeNode {
 *   int val = 0;
 *   TreeNode left = null;
 *   TreeNode right = null;
 * }
 */

public class Solution {
    /**
     * 
     * @param num int整型一维数组 
     * @return TreeNode类
     */
    public TreeNode sortedArrayToBST (int[] num) {

        if (num == null || num.length == 0) {
            return null;
        }

        return buildAvlTree(num, 0, num.length-1);
    }

    /**
     * 题目中所给的是一个升序排序的数组,所以一个大体流程就是根据一个中序遍历的序列,还原出一个它对应的二叉树。
     * 因为是中序遍历的结果,所以元素的左边都是以该元素为根节点的左子树上的元素,
     * 元素的右边都是以该元素为根节点的右子树上的元素。
     *
     * 又因为是平衡的,所以先找出序列的中点,以中点的值生成根节点,他的左边右边相差不差过一个元素,
     * 即它的左子树和右子树的元素个数相差不超过一个,每次都安一样的策略来找根节点,即左右子树的高度相差不会超过1。
     *
     * 以中点的左边序列为一个新的序列,按照一样的方法生成一个平衡二叉树,该树是第一步找的根节点的左子树。
     *
     * 以中点的右边序列为一个新的序列,按照一样的方法生成一个平衡二叉树,该树是第一步找的根节点的右子树。
     *
     * 也就是把问题的规模拆分成一个一个的小规模问题,解法都是一样的,只是问题的规模不一样,
     * 到规模足够小的时候我们可以很方便的找出答案,再一步一步往上推回去,即可得到真个大规模问题的答案。
     * 
     */
    public TreeNode buildAvlTree (int[] num, int start, int end) {

        if (start == end) {
            return new TreeNode(num[start]);
        }

        if (start > end) {
            return null;
        }

        //不能写成(start+end)/2,防止start+end越界超出int范围
        //是num的中位 所以要+1
        int mid = start + (end - start + 1)/2;

        TreeNode root = new TreeNode(num[mid]);
        root.left = buildAvlTree(num, start, mid-1);
        root.right = buildAvlTree(num, mid+1, end);

        return root;
    }
}
全部评论

相关推荐

2025-12-01 10:57
已编辑
云智研发公司_后台开发
先说明一下 bg双非本,没有特别加分的竞赛奖项,只有一些省奖作为25入职的校招生我想我我入职体验是最新鲜的面试准备,工作体验(本人是研发岗)1、首先作为一个实习经历并不多的双非本来说,我能过筛选已经很出乎意外了,所以我格外重视这次面试①首先就是算法题,我在力扣刷了两遍的hot100,最起码我认为我不能在算法题上失误②其次我在牛客网和小红书上看了很多的面经,包括一些自我介绍,一些面试技巧等tips:①简历中的内容一定要理解透彻,面试官可能问到简历中的各种内容②如果在面试中遇到不会的问题,可以直接说明,面试官可以理解校招同学我只能说足够的准备才能不浪费一次面试的机会2、入职以后,我最大的体验就是同事之间的互相帮助,大家不会一个简单的问题就不耐烦,反而会问你有没有理解,可以重复的帮助你①入职后,工作上不理解的一定要积极的询问同事或者导师或者leader,大家对于校招同学有些很高的包容性,面对其他问题可以问hr②新入职都可能存在彷徨,有压力,毕竟入职一定是需要学习新的知识,但是可以化压力为动力,努力学习3、最后我想说的,云智大家庭是一个包容性很强很温暖的大家庭,没有学历歧视,没有经验歧视,没有地域歧视,没有职位歧视,只有一起共同进步的目标,欢迎大家向云智投出简历(我本人在武汉腾讯云智,有想咨询的问题可以私聊我)ps:图一是入职培训优秀小组
面了100年面试不知...:是谁有鹅仔
腾讯云智研发成长空间 5107人发布
点赞 评论 收藏
分享
2025-12-18 22:04
已编辑
杭州电子科技大学 Java
程序员牛肉:我觉得是这样的,你现在有点病急乱投医了。你要问自己这样一个问题: 我找实习的目的是什么?为了挣钱还是增强个人实力?如果是为了挣钱那没得说,如果我是为了增强个人实习,那我异地去一个小厂实习真的有收益吗?这个收益是否大过我参加学校的项目或者自学?我记得你们杭电有那种实验室专门负责运维学校的项目的。 找实习只是一个手段而已,不要把他变成目的。不要病急乱投医。
实习简历求拷打
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务