二叉树按层遍历并收集结点

简介: 二叉树按层遍历并收集结点

二叉树按层遍历并收集结点

力扣链接:二叉树的按层遍历II

题目

给你二叉树的根节点 root ,返回其节点值 自底向上的层序遍历 。 (即按从叶子节点所在层到根节点所在的层,逐层从左向右遍历)

示例

在这里插入图片描述

提示

树中节点数目在范围 [0, 2000] 内
-1000 <= Node.val <= 1000

题解

先不关心倒序的事,这个题目返回的肯定是一个嵌套的list,外层的list套着一个个小list

在这里插入图片描述

解题步骤
  1. 拿出此时队列的size,size有多少个,步骤2重复多少回;
  2. 弹出当前结点,当前结点有左孩子就先将左孩子加入队列,有右孩子再将右孩子加入队列;(先左再右
问题

经历上面的步骤,确实是按顺序收集好了,但是题目要求的是倒序收集,即按从叶子节点所在层到根节点所在的层,逐层从左向右遍历

此时如果外层的list用的是ArrayList的话,因为它底层就是数组,所以插入数据代价是O(N)的;如果外层用的是LinkedList,它底层用的是链表,插入数据的代价就是O(1)的。

小测试(分别测试ArrayList和LinkedList插入数据的效率),每次都在0位置上插入数据,测试了300000组,单位时间为毫秒(刚兴趣的同学也可以自己测试下)

public class Code10_BinaryTreeLevelOrderTraversalII {
    public static void main(String[] args) {
        int testTimes=300000;
        long start;
        long end;
        ArrayList<Integer> arr1=new ArrayList<>();
        start=System.currentTimeMillis();
        for(int i=0; i<testTimes; i++){
            arr1.add(0,i);
        }
        end=System.currentTimeMillis();
        System.out.println(end-start);

        LinkedList<Integer> arr2=new LinkedList<>();
        start=System.currentTimeMillis();
        for(int i=0; i<testTimes; i++){
            arr2.add(0,i);
        }
        end=System.currentTimeMillis();
        System.out.println(end-start);
    }
}

测试结果
在这里插入图片描述

代码
public List<List<Integer>> levelOrderBottom(TreeNode root) {
        List<List<Integer>> ans=new LinkedList<>();
        if(root==null){
            return ans;
        }
        Queue<TreeNode> queue=new LinkedList<>();
        queue.add(root);
        while(!queue.isEmpty()){
            int size=queue.size();// 必须要记录size,不能直接 i<queue.size()
            List<Integer> curAns=new LinkedList<>();
            for(int i=0; i<size; i++){
                TreeNode curNode=queue.poll();
                curAns.add(curNode.val);
                if(curNode.left!=null){
                    queue.add(curNode.left);
                }
                if(curNode.right!=null){
                    queue.add(curNode.right);
                }
            }
            ans.add(0,curAns);
        }
        return ans;
    }

扩展

Java中队列是个接口,不能够自己实现。栈可以自己实例化,但是Java中栈的方法实现起来是比较慢的(具体为什么慢就要去看源码了,可以说是个经验),所以如果想优化常数时间的话,应该自己去实现一个栈,而别去用Java系统中提供的栈,可以用LinkedList,也可以用数组(栈不就是先进后出嘛)。

最快的情况下就是用数组实现栈(一些笔试情况下,需要优化常数时间的时候,经常用到),假设数组长度为一万,那么我们也可以确定栈的大小就是一万...

我们也可以来做个小测试

        int testTimes2=20000000;
        start=System.currentTimeMillis();
        Stack<Integer> stack1=new Stack<>();
        for(int i=0; i<testTimes2; i++){
            stack1.push(i);
        }
        while(!stack1.isEmpty()){
            int a=stack1.pop();
        }
        end=System.currentTimeMillis();
        System.out.println(end-start);


        start=System.currentTimeMillis();
        int[] stack2=new int[testTimes2];
        int i=0;
        for( ; i<testTimes2; i++){
            stack2[i++]=i;
        }
        while(i!=0){
            int b=stack2[--i];
        }
        end=System.currentTimeMillis();
        System.out.println(end-start);

测试结果:插入弹出20000000次,系统栈耗费时间6000多毫秒,自己用数组实现的栈只耗费了79毫秒,极大的优化了常数时间!
在这里插入图片描述

相关文章
|
3天前
|
数据采集 人工智能 安全
|
13天前
|
云安全 监控 安全
|
4天前
|
自然语言处理 API
万相 Wan2.6 全新升级发布!人人都能当导演的时代来了
通义万相2.6全新升级,支持文生图、图生视频、文生视频,打造电影级创作体验。智能分镜、角色扮演、音画同步,让创意一键成片,大众也能轻松制作高质量短视频。
1084 152
|
18天前
|
机器学习/深度学习 人工智能 自然语言处理
Z-Image:冲击体验上限的下一代图像生成模型
通义实验室推出全新文生图模型Z-Image,以6B参数实现“快、稳、轻、准”突破。Turbo版本仅需8步亚秒级生成,支持16GB显存设备,中英双语理解与文字渲染尤为出色,真实感和美学表现媲美国际顶尖模型,被誉为“最值得关注的开源生图模型之一”。
1751 9
|
9天前
|
人工智能 自然语言处理 API
一句话生成拓扑图!AI+Draw.io 封神开源组合,工具让你的效率爆炸
一句话生成拓扑图!next-ai-draw-io 结合 AI 与 Draw.io,通过自然语言秒出架构图,支持私有部署、免费大模型接口,彻底解放生产力,绘图效率直接爆炸。
694 152
|
11天前
|
人工智能 安全 前端开发
AgentScope Java v1.0 发布,让 Java 开发者轻松构建企业级 Agentic 应用
AgentScope 重磅发布 Java 版本,拥抱企业开发主流技术栈。
660 14
|
6天前
|
SQL 自然语言处理 调度
Agent Skills 的一次工程实践
**本文采用 Agent Skills 实现整体智能体**,开发框架采用 AgentScope,模型使用 **qwen3-max**。Agent Skills 是 Anthropic 新推出的一种有别于mcp server的一种开发方式,用于为 AI **引入可共享的专业技能**。经验封装到**可发现、可复用的能力单元**中,每个技能以文件夹形式存在,包含特定任务的指导性说明(SKILL.md 文件)、脚本代码和资源等 。大模型可以根据需要动态加载这些技能,从而扩展自身的功能。目前不少国内外的一些框架也开始支持此种的开发方式,详细介绍如下。
438 5