二叉树(Binary Tree):先序遍历、中序遍历、后序遍历和层次遍历

article/2025/9/1 18:10:21

二叉树(Binary Tree):先序遍历、中序遍历、后序遍历和层次遍历

  • 树 Tree
  • 二叉树 Binary Tree
    • 先序遍历 Preorder Traversal
    • 中序遍历 Inoreder Traversal
    • 后序遍历 Postorder Traversal
    • 层次遍历 Level Traversal

树 Tree

在这里插入图片描述

  • Root:树顶部的顶点,记为r
  • 路径 Path:树中的路径是连接一个顶点到另一个顶点的顶点和边的序列,顶点y到顶点v的路径记为 y ∗ → T v _y \ \underrightarrow{*}_T\ {_v} y  T v
  • 深度 Depth:一个顶点v的深是路径 r ∗ → T v _r \ \underrightarrow{*}_T\ {_v} r  T v上边的数目。
  • 层次 Level:从根结点开始,根结点的层次为1,根的直接后继层次为2,以此类推。
  • 节点的度:节点子树个数。
  • 父亲 Parent:如果在树上存在一个边 ( p , v ) (p,v) (p,v),则顶点p是顶点v的父亲。
  • 孩子 Child:如果在树上存在一个边 ( p , v ) (p,v) (p,v),则顶点v是顶点p的父亲。
  • 兄弟 Sibling:有相同父亲的顶点。
  • 祖先 Ancestor:如果a在路径 a ∗ → T v _a \ \underrightarrow{*}_T\ {_v} a  T v (包括v)上,则顶点a是v的祖先。
  • 后代 Descendant:顶点a是顶点v的祖先,则va的后代,包括顶点自己。
  • 真祖先 Proper Ancestor:和祖先的一样,但不包括自己,真祖先a到顶点v的路径记为: a + → T v _a \ \underrightarrow{+}_T\ {_v} a  +T v
  • 真后代 Proper descendant:和后代一样,但不包括自己。
  • 子树 Subtree:顶点v是树中的一个顶点,有v和其后代组成的的树为T的子树,顶点v是该子树的根。
  • 单体 Singlenton:只有一个顶点的树。
  • 森林 Forest:森林就是一系列的树,这些树不互相连接。
  • 叶结点 Leaf :度为0的结点称为叶结点,也可以叫做终端结点;

二叉树 Binary Tree

二叉树:所有节点的度不超过2的树。
满二叉树:一种二叉树,叶子节点都在最下面一层,并且除了叶子节点外每个节点度都为2。
完全二叉树:一种二叉树,叶子节点只能位于最下面一层或次最下面一层,并且最下面一层都位于最左边的若干子树。
在这里插入图片描述
二叉树最常使用的存储是采用链式存储:

typedef int ElemType;
class TreeNode{
private:ElemType val;TreeNode* lchild;TreeNode* rchild;
public:TreeNode(ElemType x) : val(x), child(nullptr), rchild(nullptr) {}
}

在这里插入图片描述

先序遍历 Preorder Traversal

先序遍历:先访问根节点,然后访问左子树,最后访问右子树。ABDECFG。

递归版本

void PreOrder(TreeNode* root){if(root == nullptr) return;visit(root);PreOrder(root->lchild);PreOrder(root->rchild);
}

非递归版本

借助栈来完成。

  1. 沿着左孩子访问,并依次入栈,直到左孩子为空。
  2. 栈顶元素出栈,若其右孩子为空,继续步骤2,否则对右孩子执行步骤1。
void PreOrder1(TreeNode* root){if(root == nullptr) return;Stack<TreeNode*> s;TreeNode* p = root;while(p != nullptr || !s.empty()) {if (p != nullptr) {visit(p);s.push(p);p = p->lchild;} else {p = s.pop();p = p->rchild;}}
}

中序遍历 Inoreder Traversal

中序遍历:先访问左子树,然后访问根节点,最后访问右子树。DBEAFCG。

递归版本

void InOrder(TreeNode* root){if(root == nullptr) return;PreOrder(root->lchild);visit(root);PreOrder(root->rchild);
}

非递归版本

算法和先序遍历类似,只是访问根节点时机改变了。

  1. 沿着左孩子,依次入栈,直到左孩子为空。
  2. 栈顶元素出栈并访问,若其右孩子为空,继续步骤2,否则对右孩子执行步骤1。
void InOrder1(TreeNode* root){if(root == nullptr) return;Stack<TreeNode*> s;TreeNode* p = root;while(p != nullptr || !s.empty()) {if (p != nullptr) {s.push(p);p = p->lchild;} else {p = s.pop();visit(p);p = p->rchild;}}
}

后序遍历 Postorder Traversal

后序遍历:先访问左子树,然后访问右子树,最后访问根节点。DEBFGCA。

递归版本

void PostOrder(TreeNode* root){if(root == nullptr) return;PreOrder(root->lchild);PreOrder(root->rchild);visit(root);
}

非递归版本

也借助栈:

  1. 沿着左孩子,将其依次入栈,直到左孩子为空。
  2. 读取栈顶元素(但不出栈),若其有孩子为空且未被访问过,对右孩子执行步骤1,否则,栈定元素出栈并访问。
void PostOrder1(TreeNode* root){if(root == nullptr) return;TreeNode* p = root;TreeNode* visited = nullptr;Stack<TreeNode*> s;while (p != nullptr || !s.empty()) {if (p != nullptr) {s.push(p);p = p->lchild;} else {p = s.top();if (p->rchild != nullptr && p->rchild != visited) {s.push(p->rchild);p = p->rchild->lchild;} else {p = s.pop();visit(p);visited = p;p = nullptr;  }}}
}

层次遍历 Level Traversal

层次遍历:按照层,上一层遍历完,再遍历下一层,每一层先访问左边节点,再访问右边节点。ABCDEFG。

需要借助队列,先将根节点入队,然后出队并访问。若它有左孩子则左孩子根节点入队,若它有右孩子则右孩子根节点入队。
接着出队并访问,继续上述步骤,直到队列为空。

void LevelOrder1(TreeNode* root){if(root == nullptr) return;queue<TreeNode*> q;q.push(root);while (!q.empty()) {TreeNode* p = q.front();q.pop();visite(p);if (p->lchild != nullptr) {q.push(p->lchild);}if (p->rchild != nullptr) {q.push(p->rchild);}}
}

参考:

  • [1] https://blog.csdn.net/Real_Fool_/article/details/113930623?spm=1001.2014.3001.5506
  • [2] 算法导论

http://chatgpt.dhexx.cn/article/ZAgeeDX5.shtml

相关文章

oracle awr监控报告,一个Oracle小白的AWR报告分析(一)

背景&#xff1a;某个类似准实时的数据分析系统&#xff0c;每15分钟从其他6个数据库中抽取五百张增量数据表&#xff0c;并进行15分钟粒度统计&#xff0c;同时有个前端门户进行查询。 该数据分析系统由数据抽取服务器、应用服务器、数据库服务器组成&#xff0c;全部为虚拟机…

oracle生成awr报告命令,Oracle AWR报告生成方法

1、登录Oracle程序所在的服务器&#xff0c;查找出awrrpt.sql文件所在位置 D:\oracle\product\10.2.0\db_1\RDBMS\ADMIN\awrrpt.sql 2、登录Oracle&#xff0c;以sysdba身份连接 3、执行命令 D:\oracle\product\10.2.0\db_1\RDBMS\ADMIN\awrrpt.sql 4、输入report_type报告类型…

oracle打印awr报告,oracle导出AWR报告步骤

1、进入数据库 sqlplus / as sysdba ps:如果出现用户密码错误&#xff0c; 计算机管理 > 组 > ora_dba组里的用户登陆操作系统&#xff0c;就可以无需输入用户和口令&#xff0c;直接以sysdba的身份连上数据库。 2、查看用户 show parameter db_name 3、开始压测后执行 e…

Oracle导出AWR报告

一、使用root用户登录Linux服务器 二、切换至oracle用户 执行命令&#xff1a;su – oracle&#xff0c;然后回车 三、使用管理员权限连接数据库 执行命令&#xff1a;sqlplus / as sysdba&#xff0c;然后回车 四、生成报告快照 执行脚本&#xff1a;exec DBMS_WORKLOAD_RE…

如何分析AWR 报告

Automatic Workload Repository是10g引入的一个重要组件。在里面存贮着近期一段时间内&#xff0c;默认是7天&#xff0c;数据库活动状态的详细信息。 AWR报告是对AWR视图进行查询而得到的一份自动生成的报告。可以通过下面的脚本手工得到一份AWR报告。 exec dbms_w…

oracle 取awr报告,Oracle生成awr报告

Oracle生成awr报告 达芬奇的梦 2018-04-22 21:28:32 Oracle 一、手工生成awr报告的方法 1、相应权限用户登录(sysdba)后,在$ORACLE_HOME/rdbms/admin 2、在sqlplus里执行@?/rdbms/admin/awrrpt.sql,按照提示操作。 3、生成AWR报告说明 单实例:@$ORACLE_HOME/rdbms/admin/aw…

Oracle SQL调优系列之AWR报告简介

文章目录 一、AWE报告生成步骤1.1 工具选择1.2 自动创建快照1.3 手工创建快照1.4 生成AWR报告 二、AWR报告分析2.1 AWR之DB Time2.2 AWR之load_profile2.3 AWR之efficiency percentages2.4 AWR之top 10 events2.5 AWR之SQL Statistics 一、AWE报告生成步骤 对于SQL调优&#x…

AWR报告解读

0 初步结论 ① 数据库CPU资源不够&#xff0c;CPU使用率较高&#xff0c;造成CPU等待时间较长&#xff0c;可适当提升CPU资源&#xff1b; ② 数据库I/O资源消耗不太大&#xff0c;不存在IO瓶颈&#xff1b; ③ 可适当调大SGA空间&#xff08;增加10G左右&#xff09;&#xf…

用sql统计vintage,滚动率,迁移率,逾期率

获取代码请移步&#xff1a;用sql统计vintage&#xff0c;滚动率&#xff0c;迁移率&#xff0c;逾期率

如何用R语言做Vintage分析

一、背景 Vintage一词源自葡萄酒业&#xff0c;意思是葡萄酒酿造年份。因为每年的天气、温度、湿度、病虫害等情况不同&#xff0c;而这些因素都会对葡萄酒的品质产生很大的影响&#xff0c;所以人们对葡萄酒以葡萄当年的采摘年份进行标识来加以品质区分。现在Vintage分析被广泛…

风控中必做的数据分析

大数据领域就没有不做数据分析的&#xff0c;大数据风控也不例外。 我的观点是风控和其他互联网业务都是互通的&#xff0c;本文介绍下风控中必做的数据分析&#xff0c;用以说明数据分析是一通百通的。 工欲善其事&#xff0c;必先利其器。先说下数据分析的工具。 分析工具…

Vintage、滚动率、迁移率的应用

更多风控建模、大数据分析等内容请关注公众号《bigdatafengkong》 BY 小石头 一、Vintage Vintage源于葡萄酒酿造&#xff0c;葡萄酒的品质会因葡萄生长的年份不同、气候不同而不同。Vintage分析是指评估不同年份的葡萄酒的品质随着窖藏时间的推移而发生的变化&#xff0c;并且…

窗口函数:vintage报表

0 前言 Vintage这个词原意是指酿造葡萄酒的酒窖。葡萄酒是讲究年份&#xff0c;哪年光景好&#xff0c;哪年光景不好&#xff0c;直接会影响到葡萄酒的品质。后来借用到信贷资产行业&#xff0c;指的是每个月贷款的资产质量情况&#xff0c;要直接跟每个相同时间段内的余额做比…

信贷风控中Vintage、滚动率、迁移率的理解

风控业务背景 信贷风险管理是一门艺术&#xff0c;更是一门科学。资产质量分析中常会涉及到三个理论&#xff1a; 账龄分析&#xff08;Vintage Analysis&#xff09;&#xff1a;用以分析账户成熟期、变化规律等。滚动率分析&#xff08;Roll Rate Analysis&#xff09;&#…

风控ML[9] | Vintage和Roll Rate 分析的详解

我们说了好几期的风控建模了&#xff0c;也有不少的同学私信我说一般来说我们需要怎么确定Y值呢&#xff1f;&#xff0c;到底多坏的逾期表现的客户可以被我们定义为坏客户呢&#xff1f;今天这篇文章&#xff0c;就给大家介绍一个大家既熟悉又陌生的分析工具——Vintage Analy…

了解过Vintage的N种样式?

vintage的几种形式有没有兴趣了解下&#xff1f; 我们之前写的文章里就提到过一个资产分析报表里的vintage表&#xff0c;这个表是反映客群的账龄情况&#xff0c;如果不是很清楚请再戳进去&#xff1a;风控建模系列&#xff08;六&#xff09;&#xff1a;催收评分卡卡跟贷前…

风控模型策略-知识全整理(一)

做了大概5年风控&#xff0c;中间做过甲方&#xff0c;做过乙方&#xff0c;做过模型&#xff0c;做过策略&#xff0c;做过数据分析&#xff0c;但是始终觉着不得风控精华&#xff0c;做的事情太多&#xff0c;有的东西也就很难深入&#xff0c;目前就是将这么几年的积累写下来…

vintage、滚动率等相关指标介绍

目录 1、vintage 方法简介 优势 五级分类的比较 2、滚动率 3、入催率 4、FPD 随着互联网金融的发展&#xff0c;对数据分析的需求越来越大。数据分析的目的其实是为了找到风险和收益的平衡点。高收益伴随着高风险&#xff0c;而低风险的回报又如同鸡肋。所以&#xff0c;…

对Vintage未表现数据的预测方法总结

这段时间在利用Vintage分析做借贷产品的放款损失率相关工作&#xff0c;来简单总结一下。 Vintage分析 前面说到&#xff0c;Vintage是资产质量分析的重要工具&#xff0c;主要是用来分析同一产品在不同时间放款的资产质量变化情况&#xff0c;从而反映该产品的客群质量和变化…

vintage+android相机,Vintage复古相机

Vintage复古相机是一款功能强大&#xff0c;非常好用的相机软件&#xff0c;这里有着丰富的复古滤镜可以自由选择&#xff0c;并且还可以直接在这里p图修图&#xff0c;各种效果可以提前预览&#xff0c;还可以一键生成保存&#xff0c;非常便捷&#xff01;喜欢拍照的小伙伴不…