LeetCode力扣第114题:多种算法实现 将二叉树展开为链表

作者介绍:10年大厂数据\经营分析经验,现任大厂数据部门负责人。
会一些的技术:数据分析、算法、SQL、大数据相关、python
欢迎加入社区:码上找工作
作者专栏每日更新:
LeetCode解锁1000题: 打怪升级之旅
python数据分析可视化:企业实战案例
python源码解读
程序员必备的数学知识与应用

题目描述

给定一个二叉树,原地将它展开为一个单链表。例如,给定二叉树:

    1/ \2   5/ \   \
3   4   6

展开后应该变为:

1\2\3\4\5\6

方法一:递归展开

解题步骤

  1. 如果树为空,直接返回。
  2. 递归展开左子树和右子树。
  3. 将左子树插入到右子树的位置。
  4. 将原来的右子树接到当前右子树的末端。
  5. 考虑到展开后没有左子节点,将左子节点设置为None

代码示例

class Solution:def flatten(self, root):if not root:returnself.flatten(root.left)self.flatten(root.right)# 左右子树已经被拉平成一条链表left = root.leftright = root.right# 将左子树作为右子树root.left = Noneroot.right = left# 找到当前右子树(原左子树)的末端并连接原右子树p = rootwhile p.right:p = p.rightp.right = right

方法一的改进主要可以从两个方面进行:优化递归效率和空间使用。虽然基本递归方法简单直观,但它可能导致不必要的栈空间消耗,尤其是在处理非常深的树时可能会导致栈溢出。以下是几种改进策略:

改进1: 尾递归优化

虽然Python默认不支持尾递归优化,但我们可以尽可能减少递归中不必要的操作,将必要的操作移至递归调用之前执行,减少递归调用栈的深度。

改进代码示例

class Solution:def flatten(self, root):def flattenTree(node):if not node:return None# 如果节点是叶子节点,直接返回if not node.left and not node.right:return node# 递归平展左子树leftTail = flattenTree(node.left)rightTail = flattenTree(node.right)# 将左子树插入到右子树的地方if leftTail:leftTail.right = node.rightnode.right = node.leftnode.left = None# 返回最后的尾部节点return rightTail if rightTail else leftTailflattenTree(root)
改进2: 减少递归深度

尽可能地在每个节点处理完毕后立即释放其左子树的引用,减少Python垃圾回收器的压力,并减少递归深度。

改进代码示例

class Solution:def flatten(self, root):# Helper function to perform flatten operationdef flattenNode(node):if not node:return None# Flatten the left and right subtreeleft_last = flattenNode(node.left)right_last = flattenNode(node.right)# If there is a left subtree, weave it into the right subtreeif left_last:left_last.right = node.rightnode.right = node.leftnode.left = None# The rightmost node is needed for linking to the parent node's right subtreereturn right_last if right_last else left_lastflattenNode(root)

方法二:迭代展开

解题步骤

  1. 使用栈来辅助迭代过程。
  2. 每次取出栈顶元素,如果有右子节点先压栈,再压左子节点。
  3. 修改每个节点的右指针指向下一个栈顶元素,左指针设置为None

代码示例

class Solution:def flatten(self, root):if not root:returnstack = [root]prev = Nonewhile stack:curr = stack.pop()if prev:prev.right = currprev.left = Noneif curr.right:stack.append(curr.right)if curr.left:stack.append(curr.left)prev = curr

方法三:寻找前驱节点

解题步骤

  1. 对于每个节点,如果左子节点不为空,找到左子树的最右节点(前驱节点)。
  2. 将原右子树接到前驱节点的右边。
  3. 将左子树移到右边,左子树置为空。
  4. 继续处理链表中的下一个节点。

代码示例

class Solution:def flatten(self, root):curr = rootwhile curr:if curr.left:pre = curr.leftwhile pre.right:pre = pre.rightpre.right = curr.rightcurr.right = curr.leftcurr.left = Nonecurr = curr.right

算法分析

  • 时间复杂度
    • 递归展开:(O(n)),每个节点访问一次。
    • 迭代展开:(O(n)),每个节点访问一次。
    • 寻找前驱节点:(O(n)),每个节点访问一次。
  • 空间复杂度
    • 递归展开:(O(n)),递归栈的空间。
    • 迭代展开:(O(n)),使用了额外的栈。
    • 寻找前驱节点:(O(1)),原地修改,不需要额外空间。

优劣势对比

  • 递归展开
    • 优点:实现简单直观。
    • 缺点:需要额外的栈空间来处理递归。
  • 迭代展开
    • 优点:避免了递归可能导致的栈溢出。
    • 缺点:需要使用栈来存储节点。
  • 寻找前驱节点
    • 优点:不需要额外空间,空间复杂度最优。
    • 缺点:代码逻辑相对复杂,需要处理更多的指针操作。

应用示例

这种技巧在处理需要将树结构线性化处理的场景非常有用,例如在图形界面开发中,需要按特定顺序访问或显示节点信息,或者在需要频繁查找、更新结点而又不频繁修改树结构的数据库和文件系统中。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/836515.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

目标检测YOLO实战应用案例100讲-基于深度学习的交通场景多尺度目标检测算法研究与应用(下)

目录 3.2 基于空洞卷积的特征融合模块设计 3.3 改进k-means聚类算法的anchor尺寸优化设计

微调大模型学习记录

微调大模型基本思路 一般来说, 垂直领域的现状就是大家积累很多垂域数据,从现实出发,第一步可以先做增量训练.所以会把模型分成3个阶段: (1)、第一阶段:(Continue PreTraining)增量预训练,在海量领域文档数据(领域知识)上二次预训练base模型…

Java | Leetcode Java题解之第85题最大矩形

题目&#xff1a; 题解&#xff1a; class Solution {public int maximalRectangle(char[][] matrix) {int m matrix.length;if (m 0) {return 0;}int n matrix[0].length;int[][] left new int[m][n];for (int i 0; i < m; i) {for (int j 0; j < n; j) {if (mat…

Python3 + Appium + 安卓模拟器实现APP自动化测试并生成测试报告

这篇文章主要介绍了Python3 Appium 安卓模拟器实现APP自动化测试并生成测试报告,本文给大家介绍的非常详细&#xff0c;对大家的学习或工作具有一定的参考借鉴价值&#xff0c;需要的朋友可以参考下 本文主要分为以下几个部分 安装Python3 安装Python3的Appium库 安装Andr…

Mp3tag for Mac:音乐标签,轻松管理

还在为杂乱无章的音乐文件而烦恼吗&#xff1f;Mp3tag for Mac&#xff0c;让您的音乐库焕然一新&#xff01;它支持多种音频格式&#xff0c;批量编辑标签&#xff0c;让音乐管理变得简单高效。同时&#xff0c;自动获取在线数据库的音乐元数据&#xff0c;确保您的音乐库始终…

优选算法——双指针

优选算法 一、leetcode283.移动零 题目分析&#xff1a; ​ 给定一个数组 nums&#xff0c;编写一个函数将所有 0 移动到数组的末尾&#xff0c;同时保持非零元素的相对顺序。 算法原理&#xff1a; ​ 快排原理&#xff1a; ​ 数组划分或者数组分块&#xff1a;即在一定…

docker 开启 tcp 端口

前言&#xff1a;查了很多网上资料 都说要修改daemons,json 完全不管用&#xff0c;而且还导致添加 {“host”:["tcp://0.0.0.0:2375","unix:///var/lib/docker.sock"]} 后&#xff0c;docker restart 失败&#xff0c;浪费了不少时间 &#xff01;&am…

基于物理的渲染的光照参数

基于物理的渲染通常使用更加真实和复杂的光照模型&#xff0c;其中包括一系列物理参数来模拟光线在场景中的传播和反射。以下是一些常见的基于物理的渲染光照参数&#xff1a; 环境光&#xff08;Ambient Light&#xff09;&#xff1a;环境光在基于物理的渲染中仍然存在&#…

kafka安装配置及集成springboot

1. 安装 单机安装kafka Kafka对于zookeeper是强依赖&#xff0c;保存kafka相关的节点数据&#xff0c;所以安装Kafka之前必须先安装zookeeper dockerhub网址: https://hub.docker.com Docker安装zookeeper 下载镜像&#xff1a; docker pull zookeeper:3.4.14创建容器 doc…

docker(五):DockerFile

文章目录 DockerFile1、Dockerfile构建过程解析2、DockerFile常用保留字命令FROMMAINTAINERRUNEXPOSEWORKDIRUSERENVADDCOPYVOLUMECMDENTRYPOINT总结 3、案例 DockerFile 1、Dockerfile构建过程解析 官网文档&#xff1a;https://docs.docker.com/reference/dockerfile/ Dock…

【论文阅读笔记】HermesSim(Code is not Natural Language) (Security 24)

个人博客地址 HermesSim [Security 24] 论文&#xff1a;《Code is not Natural Language: Unlock the Power of Semantics-Oriented Graph Representation for Binary Code Similarity Detection》 仓库&#xff1a;https://github.com/NSSL-SJTU/HermesSim 提出的问题 二…

JVM调优:JVM中的垃圾收集器详解

JVM&#xff08;Java Virtual Machine&#xff09;垃圾收集器是Java虚拟机中的一个重要组件&#xff0c;负责自动管理Java堆内存中的对象。垃圾收集器的主要任务是找出那些不再被程序使用的对象&#xff0c;并释放它们占用的内存&#xff0c;以便为新的对象分配空间。这个过程被…

C#泛型委托

在C#中&#xff0c;delegate 关键字用于声明委托&#xff08;delegates&#xff09;&#xff0c;委托是一种类型安全的函数指针&#xff0c;允许你传递方法作为参数或从方法返回方法。有时我们需要将一个函数作为另一个函数的参数&#xff0c;这时就要用到委托&#xff08;Dele…

算法题② —— 链表专栏

1. 链表数据结构 struct ListNode {int val;ListNode *next;ListNode() : val(0), next(nullptr) {}ListNode(int x) : val(x), next(nullptr) {}ListNode(int x, ListNode *next) : val(x), next(next) {}};2. 链表的删除 2.1 移除链表元素 力扣&#xff1a;https://leetco…

常见的自定义类加载器实现策略

1、从特定路径加载类&#xff1a; 这种策略允许类加载器从文件系统上的特定目录或JAR文件中加载类。例如&#xff0c;可以将应用程序的插件或扩展放在特定的目录中&#xff0c;并使用自定义类加载器动态加载它们。 2、从网络加载类&#xff1a; 自定义类加载器可以从远程服务…

引擎:主程渲染

一、引擎发展 二、引擎使用 1.游戏渲染流程 2.3D场景编辑器操作与快捷键 3.节点的脚本组件 脚本介绍 引擎执行流程 物体节点、声音组件\物理组件\UI组件、脚本组件 暴露变量到面板 4.节点的查找 基本查找 this.node&#xff1a;挂载当前脚本的节点A&#xff1b; this.nod…

Git LFS拉取大文件

当你使用Git LFS来管理大文件时&#xff0c;拉取包含这些大文件的仓库会稍微有些不同&#xff0c;但大部分过程自动化&#xff0c;用户体验类似于正常的Git操作。以下是拉取包含Git LFS对象的仓库的步骤&#xff1a; 确保已安装 Git LFS 在你尝试拉取任何包含通过Git LFS跟踪…

git两个不同仓库代码的同步

应用场景 将A仓库中代码同步到B仓库&#xff0c;并且在B仓库能够更新A仓库中的代码。 解决方案 解决步骤&#xff1a; 在仓库B中 首先&#xff0c;使用 git remote add 命令将 A 仓库添加为 B 仓库的远程仓库。确保你使用正确的远程仓库 URL。 git remote add repository_A…

一、精准化测试介绍

精准化测试介绍 一、精准化测试是什么&#xff1f;二、什么是代码插桩&#xff1f;三、两种插桩方式Offine模式&#xff1a;On-the-fly插桩: 四、jacoco覆盖率报告展示五、增量代码覆盖率监控原理六、精准测试系统架构图七、全量与增量覆盖率报告包维度对比八、全量与增量覆盖率…

Python 之 Flask框架(进阶版)

前面说了flask框架的基础用法&#xff0c;现在说一下Flask框架的进阶用法:#包引入 from flask import Flask,redirect,url_for,render_template,request,make_response,send_file,abort,flash, get_flashed_messages,Blueprint,views from os import urandomapp Flask(__name_…