Python - 深夜数据结构与算法之 Tree

news2025/1/11 11:56:40

目录

一.引言

二.树与二叉树简介

1.Tree 树

2.Binary Tree 二叉树

3.Binary Search Tree 二叉搜索树

三.经典算法实战

1.In-Order-Traversal [94]

2.Pre-Order-Traversal [144]

3.Fib [509]

4.N-Tree-Pre-Order-Traversal [589]

5.N-Tree-Post-Order-Traversal [590]

四.总结


一.引言

本期我们介绍常用的数据结构树以及其常用的表现形式: 二叉树、二叉搜索树。介绍相关算法前,我们先熟悉下树的结构以及其特点。

二.树与二叉树简介

1.Tree 树

介绍树之前,我们回顾一下之前学习过的数据结构 - LNode 链表,对于每个 Node 而言其只有一个 Next 指针,由此带来的后果是当我们需要遍历链表中某个元素时我们需要 o(n) 的时间复杂度,这个时候出现了跳表,调表通过引入二级、三级、... 多级索引,将数据结构升维,提高其查找元素的时间复杂度。这里 Tree 树的思想就是对一维数据结构的升维,对于某个节点 Node,其 next 指针不在限定为 1 个,而是修改为多个 next 指针,如果是 2 个那就是二叉树、如果是多个那就是多叉树。

其拥有多个属性:

Root - 根节点,作为整棵树的起始点

Parent/Child Node - 父节点、子节点,例如 D-H 就是父子节点,但 D 又是 B 的子节点

Slidings - 兄弟节点、也可以说姐妹节点,可以理解为一个 Node 下的多个 next 指针,同一级

Sub-Tree - 子树,对于 DHI 构成的子树而言,D 就变成了子树的 root 节点

Level - 层级,树一般是多级结构,不同层又叫树的深度

  

2.Binary Tree 二叉树

二叉树的儿子节点最多只有两个,即左子结点和右子节点。

◆ 代码实现

◆ 遍历方式

由于树结构的原因,我们不太好使用 For 循环来遍历,除非使用 BFS 或者堆的形式去遍历每一层的数组,基于其 Node 以及 Next 的特点,我们可以通过递归来实现树的遍历。

遍历代码

下面是 python 的示例代码,其遵循上面遍历的规则:

3.Binary Search Tree 二叉搜索树

上面提到的二叉树想要查找一个元素依旧需要遍历 o(n) 时间复杂度,因为需要遍历每个节点,这样做其实和链表没有太大区别,为了提高搜索元素的效率,我们可以让二叉树的数据变得更加有序,从而提高搜索效率。 我们将具备如下性质的树称为二叉搜索树:

注意这里中序遍历是 '根左右',由于左节点都小于根,右节点都大于根,所以二叉搜索树的中序遍历是升序排列。 

二叉搜索树搜索、插入的时间复杂度均为 o(logn),虽然相比数组 o(1) 的插入复杂度高了一些,但其 o(logn) 的搜索复杂度会比 o(n) 少很多,尤其是当 n 非常的大的时候。

◆ 搜索

◆ 插入

在上面二叉搜索树的基础上插入 26:

在上面二叉搜索树的基础上插入 55:

◆ 删除 

在上面二叉搜索树的基础上删除 32:

在上面二叉搜索树的基础上删除 65:

删除叶子节点相对容易,直接删除即可,但是如果删除子树或者树的根节点会涉及到树形态的重构,一般而言,删除树节点一般会选取 Node 的右子树最左边的节点即 72,将 72 替换到 65 的位置并重新调整。

这里继续删除 41 也是同理,我们找到 41 右子树中最左边即最小的节点进行替换即 50:

替换后如下状态:

◆ 特殊状态

二叉搜索树添加、删除、查找的时间复杂度均为 o(logn),特殊情况下退化为 o(n),下面就是一种典型的特殊情况,其从树结构退化为链表。

可视化界面: https://visualgo.net/zh/bst

三.经典算法实战

1.In-Order-Traversal [94]

二叉树的中序遍历: https://leetcode.cn/problems/binary-tree-inorder-traversal/

◆ 题目分析

二叉树的前中后序遍历都是很经典的算法,很多同学总是记不清口诀,这里博主分享下自己的记忆方法,不管什么情况下左右都是固定死的,只有根节点 root 是随着遍历方法不同而改变。前序遍历- 根在前,根左右;中序遍历-根在中,左根右;后续遍历-根在后,左右根,所以我们只需要把左中右对应到 root,左右不动即可。

◆ 中序遍历

# Definition for a binary tree node.
# class TreeNode(object):
#     def __init__(self, val=0, left=None, right=None):
#         self.val = val
#         self.left = left
#         self.right = right
class Solution(object):
    def inorderTraversal(self, root):
        """
        :type root: TreeNode
        :rtype: List[int]
        """
        if not root:
            return []
        return self.inorderTraversal(root.left) + [root.val] + self.inorderTraversal(root.right)

直接左根右遍历即可,注意判断 root 为空的条件,如果是前序和后序,只需要更换 [root.val] 的位置即可,这里三种遍历方式是类似的写法。

◆ ​​​​​​​栈实现

# Definition for a binary tree node.
# class TreeNode(object):
#     def __init__(self, val=0, left=None, right=None):
#         self.val = val
#         self.left = left
#         self.right = right
class Solution(object):
    def inorderTraversal(self, root):
        """
        :type root: TreeNode
        :rtype: List[int]
        """
        if not root:
            return []


        # 存储结果 & 栈
        re = []
        stack = []

        cur = root

        # 起始情况 cur 非空 stack 空;后续情况 cur 可能空、栈非空
        while stack or cur:
            # 中序遍历 - 左根右,所以需要找到最左边
            while cur:
                stack.append(cur)
                cur = cur.left
            
            node = stack.pop()
            re.append(node.val)

            if node.right:
                cur = node.right

        return re

上面代码不清晰的同学可以参考下面的示意图,我们通过第二个 while 保证在任何时候都找到最左侧的点, 随后依次弹出,如果右节点非空,则进入右子树。

2.Pre-Order-Traversal [144]

前序遍历: https://leetcode-cn.com/problems/binary-tree-preorder-traversal/

◆ ​​​​​​​题目分析

和前面的中序遍历基本类似,这里我们巩固一下,左右不动,动 root,所以前序遍历是 '根左右'。

◆ ​​​​​​​递归实现

class Solution(object):
    def preorderTraversal(self, root):
        """
        :type root: TreeNode
        :rtype: List[int]
        """
        if not root:
            return []

        return [root.val] + self.preorderTraversal(root.left) + self.preorderTraversal(root.right)

这个写法和上面基本相同,下面我们再尝试一种新的写法,我们的递归实际是程序自己生成了一个调用栈,下面我们自己生成栈遍历,顺便复习一下前面的 Stack。 

◆ ​​​​​​​栈实现

# Definition for a binary tree node.
# class TreeNode(object):
#     def __init__(self, val=0, left=None, right=None):
#         self.val = val
#         self.left = left
#         self.right = right
class Solution(object):
    def preorderTraversal(self, root):
        """
        :type root: TreeNode
        :rtype: List[int]
        """

        # 异常判断
        if not root:
            return []
        
        # 存放结果
        res = []

        # 显式维护一个 Stack
        stack = [root]

        # 前序遍历为根左右, 这里栈是先进后出,所以右先进左后进,最后左先出
        while stack:
            node = stack.pop()
            if node:
                res.append(node.val)

                if node.right:
                    stack.append(node.right)
            
                if node.left:
                    stack.append(node.left)
        
        return res
                

3.Fib [509]

斐波那契数列: https://leetcode.cn/problems/fibonacci-number

◆ ​​​​​​​题目分析

斐波那契数列是后项等于前两项之和,即 F(n) = F(n-1) + F(n-1),这个结构可以看作是一颗 subtree 子树,root 为 F(n),左右子节点为 F(n-1) 和 F(n-2),针对 F(n) 我们只需一直向下分叉,找到有多少个叶子节点即可。

◆ ​​​​​​​向下递归

class Solution(object):
    def fib(self, n):
        """
        :type n: int
        :rtype: int
        """
        # 0 和 1 的边界情况,防止越界
        if n == 0:
            return 0
        elif n == 1:
            return 1
        else:
            # 向下递归
            return self.fib(n-1) + self.fib(n-2)

这里可以再简化一下,上面 n=0 return 0、n=1 return 1,可以改成 if n <= 1 return n,这样两行代码就搞定了,不过执行用时有点恼火。 

◆ ​​​​​​​交替前进

class Solution(object):
    def fib(self, n):
        """
        :type n: int
        :rtype: int
        """
        a, b = 0, 1

        # 向右移动
        for i in range(n):
            a , b = b, a+b
        
        return a

这个写法也是基于斐波那契数列的性质进行的,我们只需要维护三元组即可 [a, b, a+b],然后一直更新 a,b 的指针向右移动,直到我们需要的 n。 

4.N-Tree-Pre-Order-Traversal [589]

N 叉树的前序遍历: https://leetcode-cn.com/problems/n-ary-tree-preorder-traversal/

◆ ​​​​​​​题目分析

二叉树的前序遍历是根左右,固定左右,根在前,多叉树根在前,左右变成了从左到右遍历。

◆ ​​​​​​​前序遍历​​

"""
# Definition for a Node.
class Node(object):
    def __init__(self, val=None, children=None):
        self.val = val
        self.children = children
"""

class Solution(object):
    def preorder(self, root):
        """
        :type root: Node
        :rtype: List[int]
        """
        re = []

        def dfs(node):
            if not node:
                return []
            re.append(node.val)
            for ch in node.children:
                dfs(ch)
        
        dfs(root)
        return re

和下面的多叉树的后序遍历思路一致,就是换了添加 val 的顺序。

5.N-Tree-Post-Order-Traversal [590]

N 叉树的后序遍历: https://leetcode-cn.com/problems/n-ary-tree-postorder-traversal/

◆ ​​​​​​​题目分析

二叉树后序遍历,固定左右,根放在最后,这里多叉树不止多个叉,但是根依然在最后,所以变成了左 -> 右 + 根,所以增加 for 循环从左到右遍历即可。

◆ ​​​​​​​后序遍历

"""
# Definition for a Node.
class Node(object):
    def __init__(self, val=None, children=None):
        self.val = val
        self.children = children
"""

class Solution(object):

    def postorder(self, root):
        """
        :type root: Node
        :rtype: List[int]
        """

        re = []

        def dfs(node):
            if not node:
                return []
            # 左右根,这里左右变多了,但是从左到右的顺序不变,左->右,再根
            for ch in node.children:
                dfs(ch)
            re.append(node.val)
        
        dfs(root)
        return re


        
        

四.总结

上面讲解了 Tree 树、Binary Tree 二叉树以及 Any Tree 多叉树的基本概念以及 LeetCode 里一些基本的算法操作,主要了解了树的结构特点,严格意义上说,栈是特殊的树,二叉树是特殊的多叉树。本节的题目中我们着重掌握递归的思想,因为树的结构不好显式的进行遍历,其次由于递归内部采用 Stack 实现,因此最好也要掌握手动栈实现树的遍历。最好感慨一下这些题目经常做,但是经常忘,就像赵本山测试范伟智商似的,只能答老题,而且老题稍微一变花样就又被忽悠了,哎...多多练习吧,啥时候练得过拟合就好了。

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

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

相关文章

改变传媒格局的新趋势

在如今信息高速发展的时代&#xff0c;人们早已进入了一个以手机为中心的智能化时代。随着科技的迅猛发展&#xff0c;手机无人直播成为了一种新兴的传媒形态&#xff0c;正逐渐改变着传媒格局。本文将从手机无人直播的定义、发展背景和影响等方面进行探讨。 首先&#xff0c;…

浏览器缓存机制(详)

目录 1&#xff0c;缓存的分类1.1&#xff0c;按缓存位置1&#xff0c;Service Worker2&#xff0c;Memory Cache3&#xff0c;Disk Cache4&#xff0c;Push Cache 1.2&#xff0c;按缓存类型强缓存ExpiresCache-control 协商缓存Last-Modified & If-Modified-SinceEtag &a…

【优化】XXLJOB修改为使用虚拟线程

【优化】XXLJOB修改为使用虚拟线程 新建这几个目录 类&#xff0c; 去找项目对应的xxljob的源码 主要是将 new Thread 改为 虚拟线程 Thread.ofVirtual().name("VT").unstarted 以下代码是 xxljob 2.3.0版本 举一反三 去修改对应版本的代码 <!-- 定…

UG螺旋线命令的使用

螺旋线按照螺距类型可以分为两种类型&#xff1a; 1、等螺距螺旋线 2、变螺距螺旋线 等螺距螺旋线 变螺距螺旋线 沿矢量螺旋线 沿矢量螺旋线-线性大小和螺距 沿矢量螺旋线-沿脊线的线性 沿矢量螺旋线-沿脊线的线性 当我们想模拟弹簧被拉伸或压缩状态时&#xff0c;可以使用…

Python-基于fastapi实现SSE流式返回(类似GPT)

最近在做大模型对话相关功能&#xff0c;需要将对话内容流式返回给前端页面&#xff08;类似GPT的效果&#xff09;。下面直接说下如何实现&#xff1a; 1.首先导入fastapi和sse流式返回所需要的包 from fastapi import APIRouter, Response, status from sse_starlette.sse …

智能化运输与航空航天:发展历程、问题与未来趋势

导言 智能化运输与航空航天是当前科技领域的研究热点之一&#xff0c;本文将深入研究这一领域的发展历程、遇到的问题、解决过程&#xff0c;以及未来的可用范围。同时&#xff0c;我们将探讨各国在这一领域的应用情况和未来的研究趋势&#xff0c;分析在哪些方面能够取胜&…

Java操作Word修订功能:启用、接受、拒绝、获取修订

Word的修订功能是一种在文档中进行编辑和审阅的功能。它允许多个用户对同一文档进行修改并跟踪这些修改&#xff0c;以便进行审查和接受或拒绝修改。修订功能通常用于团队合作、专业编辑和文件审查等场景。 本文将从以下几个方面介绍如何使用免费工具Free Spire.Doc for Java在…

数据挖掘体系介绍

数据挖掘是什么&#xff1f; 简而言之&#xff0c;对数据进行挖掘&#xff0c;从中提取出有效的信息。一般我们会把这种信息通过概念、规则、规律、模式等有组织的方式展示出来&#xff0c;形成所谓的知识。特别是在这个大数据时代&#xff0c;当数据多到一定程度&#xff0c;…

shell 数组的详细用法

简介 数组是一种数据结构&#xff0c;用于存储和处理一组相关的数据元素。数组可以包含多个值&#xff0c;每个值都有一个索引&#xff0c;用于标识和访问它们。 目录 1. 数组的基本用法 1.1. 定义数组的方式 1.1.1. 直接赋值 1.1.2. declare声明数组 1.1.3. 索引赋值 1.…

山景DU561—32位高性能音频处理器(DSP)芯片

音频处理可以更好地捕捉和处理声音和音乐&#xff1b;而DSP音频处理芯片是一种利用数字信号处理技术进行音频处理的专用芯片&#xff1b;可用于多种应用&#xff0c;从音乐拾音到复杂的音频信号处理&#xff0c;和声音增强。 由工采网代理的山景DU561是一款集成多种音效算法高…

YOLOv5改进 | 卷积篇 | 通过RFAConv重塑空间注意力(深度学习的前沿突破)

一、本文介绍 本文给大家带来的改进机制是RFAConv&#xff0c;全称为Receptive-Field Attention Convolution&#xff0c;是一种全新的空间注意力机制。与传统的空间注意力方法相比&#xff0c;RFAConv能够更有效地处理图像中的细节和复杂模式(适用于所有的检测对象都有一定的…

CentOS安装Python解释,CentOS设置python虚拟环境,linux设置python虚拟环境

一、安装python解释器 1、创建解释器安装的目录&#xff1a;/usr/local/python39 cd /usr/local mkdir python39 2、下载依赖 yum -y install zlib-devel bzip2-devel openssl-devel ncurses-devel sqlite-devel readline-devel tk-devel gcc make libffi-devel xz-devel …

MyBatis——MyBatis的ORM映射和MyBatis的配置文件升级

1.MyBatis的ORM映射 拷贝之前的工程&#xff1a; 1.1.什么是ORM映射 MyBatis只能自动维护库表”列名“与”属性名“相同时的对应关系&#xff0c;二者不同时无法自动ORM&#xff0c;如下&#xff1a; 1.2.列的别名 在SQL中使用 as 为查询字段添加列别名&#xff0c;以匹配…

Gin之GORM事务(转账操作)

禁用默认事务的操作 为了确保数据一致性,GORM 会在事务里执行写入操作(创建、更新、删除)。如果没有这方面的要求,您可以在初始化时禁用它,这将获得大约 30%+ 性能提升。 // 全局禁用 db, err := gorm.Open(sqlite.Open("gorm.db"), &gorm.Config{SkipDef…

CCF编程能力等级认证GESP—C++6级—20230923

CCF编程能力等级认证GESP—C6级—20230923 单选题&#xff08;每题 2 分&#xff0c;共 30 分&#xff09;判断题&#xff08;每题 2 分&#xff0c;共 20 分&#xff09;编程题 (每题 25 分&#xff0c;共 50 分)小杨买饮料小杨的握手问题 答案及解析单选题判断题编程题1编程题…

FAVDICE - Favorite Dice

题意&#xff1a;n个面的骰子&#xff0c;问期望骰多少次可以将所有n个面都骰到 思路&#xff1a;期望dp 状态表示&#xff1a;dp[i]代表已经骰出了i个面&#xff0c;还需要期望骰dp[i]次才能将n个面都骰到 状态转移&#xff1a;对于dp[i]我们考虑两种情况&#xff1a; 1、…

python flask+vue实现前后端图片上传

python flaskvue实现前后端图片上传 vue代码如下&#xff1a; <template><div><input type"file" change"handleFileChange"/><button click"uploadFile">上传</button><br><img :src"imageUrl&…

使用python免费调用Google发布的Gemini双子座大模型API

上期文章,我们介绍了google发布的Gemini双子座大模型,现在google开放了gemini-pro与gemini-pro- vision2个版本的API接口。 其中gemini-pro模型类似与ChatGPT,是一个文本输入输出聊天模型,而vision模型,顾名思义是一个多模态模型,可以支持图片与文本的输入。 我们进入如…

指定每个子字符串长度L将字符串S分割为多个长度最长为L的子字符串textwrap.wrap(s,L)

【小白从小学Python、C、Java】 【计算机等考500强证书考研】 【Python-数据分析】 指定每个子字符串长度L 将字符串S分割为多个 长度最长为L的子字符串 textwrap.wrap(s,L) [太阳]选择题 请问以下代码最后输出的结果是&#xff1f; import textwrap a "You are…

OpenCV-9颜色空间的转换

颜色转换API&#xff1a;cvtColor&#xff08;img&#xff0c;colorsapce&#xff09; cvt含义为转换 convesion(转换) 下面为示例代码&#xff1a; import cv2# callback中至少有一个参数 def callback(value):passcv2.namedWindow("color", cv2.WINDOW_NORMAL) …