初识 二叉树

目录

  • 什么是二叉树
    • 二叉树的五种状态
    • 满二叉树
    • 完全二叉树
    • 二叉排序树
    • 平衡二叉树
  • 二叉树的遍历
    • B3642 二叉树的遍历
    • P1305 新二叉树
  • 二叉树的深度
    • P4913 【深基16.例3】二叉树深度
  • 相关例题训练:
    • 二叉树问题

在这里插入图片描述
这是树(拍摄于郑州轻工业大学,第一次郑州轻工业新生赛~)
这是树的一些概念:
在这里插入图片描述

什么是二叉树

???

二叉树是n(n>=0)个节点的有限集合。

  • 1.每个节点最多只有两个子树
  • 2.左右子树不能颠倒
    (二叉树是有序树)

二叉树的五种状态

在这里插入图片描述

几种特殊的二叉树:

满二叉树

高度为h,且含有2h2^h2h-1个结点的二叉树
特点:

  • 1.只有最后一层有叶子结点
  • 2.不存在度为一的点
  • 3.按层序从1开始编号结点i的左孩子为2i,右孩子为2i+1
    在这里插入图片描述

完全二叉树

当且仅当其每个结点都与高度为h的满二叉树中编号问为1~n的结点一 一对应时成为完全二叉树。
特点

  • 1.只有最后两层可能有叶子结点。
  • 2.最多 只有一个度为1的结点。
  • 3.按层序从1开始编号结点i的左孩子为2i,右孩子为2i+1
    在这里插入图片描述

二叉排序树

左子树上所有结点的关键字均小于根节点的关键字
右子树上所有结点的关键字均大于根节点的关键字

左子树和右子树又分别时一颗二叉排序树
在这里插入图片描述

平衡二叉树

树上任一结点的左子树和右子树的深度之差不超过1
(有更高的搜索效率)
在这里插入图片描述

二叉树的遍历

前序遍历
在这里插入图片描述
中序遍历
在这里插入图片描述
后序遍历
在这里插入图片描述
关于遍历二叉树,有一个巧妙的方法分享给大家。
以下图为例:
在这里插入图片描述
中序遍历左根右 为例:
我们可以先遍历最上边的ABC, 并给B和C的子节点留上位置
_ B _ A _ C _
然后再将B和C的子节点按左根右的顺序填上去
就是这个顺序:DBEAFCG
同理,你可以练习一下:
先序遍历ABDECFG
后序遍历DEBFGCA

有了以上的基础,我们拿道题练练手吧!

B3642 二叉树的遍历

B3642 二叉树的遍历

题目描述

有一个 n(n≤106)n(n \le 10^6)n(n106) 个结点的二叉树。给出每个结点的两个子结点编号(均不超过 nnn),建立一棵二叉树(根节点的编号为 111),如果是叶子结点,则输入 0 0

建好树这棵二叉树之后,依次求出它的前序、中序、后序列遍历。

输入格式

第一行一个整数 nnn,表示结点数。

之后 nnn 行,第 iii 行两个整数 lllrrr,分别表示结点 iii 的左右子结点编号。若 l=0l=0l=0 则表示无左子结点,r=0r=0r=0 同理。

输出格式

输出三行,每行 nnn 个数字,用空格隔开。

第一行是这个二叉树的前序遍历。

第二行是这个二叉树的中序遍历。

第三行是这个二叉树的后序遍历。

输入 #1

7
2 7
4 0
0 0
0 3
0 0
0 5
6 0

输出 #1

1 2 4 3 7 6 5
4 3 2 1 6 5 7
3 4 2 5 6 7 1

这是一道很模板的二叉树遍历练习题,很适合新手宝宝体质,按顺序根据前序中序和后续的遍历顺序,结合深搜就可以很容易的输出顺序啦~代码注释很详细!

AC代码

#include<bits/stdc++.h>
using namespace std;
const int N=1e6+6;
int n,l[N],r[N];//l和r分别存左子节点和右子节点
//前序遍历,根左右  
void a(int x)//前序遍历访问到第x号点 
{if(x==0)return ;//题目中说这个结点为0时表示无此结点//然后就是按照前序遍历cout<<x<<" ";//先输出根a(l[x]);//再输出左子结点a(r[x]);//最后输出右子节点	
} 
//中序遍历,左根右
void b(int x)//中序遍历访问到第x号点 
{if(x==0)return ;//中序遍历 b(l[x]);//先输出左子结点cout<<x<<" ";//再输出根b(r[x]);//最后输出右子节点	
} 
//后序遍历,左右根
void c(int x)//后序遍历访问到第x号点 
{if(x==0)return ;//后序遍历顺序 c(l[x]);//先输出左子结点c(r[x]);//再输出右子节点	cout<<x<<" ";//最后输出根
} 
int main()
{cin>>n;for(int i=1;i<=n;i++)cin>>l[i]>>r[i];//输入对应位置的左右节点 //前序遍历,根左右 a(1);//根节点从1开始遍历cout<<endl;//前序遍历完后要输出换行//中序遍历,左根右b(1);//根节点也是从1开始中序遍历cout<<endl;//后序遍历,左右根c(1);cout<<endl; return 0;
} 

再来一道例题练练手吧!

P1305 新二叉树

P1305 新二叉树

题目描述

输入一串二叉树,输出其前序遍历。

输入格式

第一行为二叉树的节点数 nnn。(1≤n≤261 \leq n \leq 261n26)

后面 nnn 行,每一个字母为节点,后两个字母分别为其左右儿子。特别地,数据保证第一行读入的节点必为根节点。

空节点用 * 表示

输出格式

二叉树的前序遍历。

输入 #1

6
abc
bdi
cj*
d**
i**
j**

输出 #1

abdicj

一道很基础的二叉树题,可以通过结构体将这个二叉树建立起来,虽然题目中给的字符,但同样可以存储在结构体数组中,因为字符ACS码最大不超过128,所以数组只需开150就足够,然后可以利用深搜,将第一个节点传入dfs,依次搜索,当子节点不为 * 时才继续往下搜。

AC代码

#include<bits/stdc++.h>
using namespace std;
#define IOS ios::sync_with_stdio(0),cin.tie(0),cout.tie(0)
#define int long long
#define PII pair<int,int>
#define fi first
#define se second
#define endl '\n'
const int N=1e6+6;
struct node//简单的建树 
{char l,r;
}p[150];
int n;
void dfs(char bg)
{cout<<bg;if(p[bg].l !='*') dfs(p[bg].l);//如果不为空节点就接着往下搜 if(p[bg].r !='*') dfs(p[bg].r);
}
void solve()
{cin>>n;char a,x,y,bg;cin>>a>>x>>y;bg=a;//作为初始深搜的点 p[a].l =x,p[a].r =y;//左右子数 n-=1;while(n--){cin>>a>>x>>y;p[a].l =x,p[a].r =y;}dfs(bg);
}
signed main()
{IOS;int _=1;
//	cin>>_;while(_--)solve();return 0;
}

二叉树的深度

二叉树深度简而言之就是这个二叉树最多有几层
在这里插入图片描述
比如这个二叉树,它的深度就是3

我们直接上例题感受一下吧!

P4913 【深基16.例3】二叉树深度

题目描述

有一个 n(n≤106)n(n \le 10^6)n(n106) 个结点的二叉树。给出每个结点的两个子结点编号(均不超过 nnn),建立一棵二叉树(根节点的编号为 111),如果是叶子结点,则输入 0 0

建好这棵二叉树之后,请求出它的深度。二叉树的深度是指从根节点到叶子结点时,最多经过了几层。

输入格式

第一行一个整数 nnn,表示结点数。

之后 nnn 行,第 iii 行两个整数 lllrrr,分别表示结点 iii 的左右子结点编号。若 l=0l=0l=0 则表示无左子结点,r=0r=0r=0 同理。

输出格式

一个整数,表示最大结点深度。

输入 #1

7
2 7
3 6
4 5
0 0
0 0
0 0
0 0

输出 #1

4

思路分析

我们可以先利用结构体读入这个二叉树

  • 拥有左子节点和右子节点两个参数的结构体
  • 开n范围的结构体数组

搜索(dfs)

  • 状态:当前走到什么编号的节点以及当前的深度
  • 终止条件:走到0号节点(更新最大深度)
  • 走到哪里去?当前所在编号的节点的左右子节点

输出最大深度

AC代码

#include<bits/stdc++.h>
using namespace std;
#define IOS ios::sync_with_stdio(0),cin.tie(0),cout.tie(0)
#define int long long
#define PII pair<int,int>
#define fi first
#define se second
#define endl '\n'
const int N=1e6+6;
struct node//建树 
{int l,r;
}p[N];
int n,ans=INT_MIN;//ans用来记录树的最大深度 
void dfs(int x,int h)
{//终止条件:子节点为0时 ans=max(ans,h);//更新最大值 //走到哪里去if(p[x].l)//如果左子节点不为0 dfs(p[x].l,h+1);if(p[x].r)//如果右子节点不为0 dfs(p[x].r,h+1); 
}
void solve()
{cin>>n;for(int i=1;i<=n;i++)cin>>p[i].l>>p[i].r;dfs(1,1);//传入最初所在位置和最初深度cout<<ans; 
}
signed main()
{IOS;int _=1;
//	cin>>_;while(_--)solve();return 0;
}

相关例题训练:

二叉树问题

P3884 [JLOI2009] 二叉树问题
题目描述

如下图所示的一棵二叉树的深度、宽度及结点间距离分别为:

  • 深度:444
  • 宽度:444
  • 结点 8 和 6 之间的距离:888
  • 结点 7 和 6 之间的距离:333

其中宽度表示二叉树上同一层最多的结点个数,节点 u,vu, vu,v 之间的距离表示从 uuuvvv 的最短有向路径上向根节点的边数的两倍加上向叶节点的边数。

给定一颗以 1 号结点为根的二叉树,请求出其深度、宽度和两个指定节点 x,yx, yx,y 之间的距离。

输入格式

第一行是一个整数,表示树的结点个数 nnn
接下来 n−1n - 1n1 行,每行两个整数 u,vu, vu,v,表示树上存在一条连接 u,vu, vu,v 的边。
最后一行有两个整数 x,yx, yx,y,表示求 x,yx, yx,y 之间的距离。

输出格式

输出三行,每行一个整数,依次表示二叉树的深度、宽度和 x,yx, yx,y 之间的距离。

输入 #1

10                                
1 2                            
1 3                            
2 4
2 5
3 6
3 7
5 8
5 9
6 10
8 6

输出 #1

4
4
8

说明/提示

对于全部的测试点,保证 1≤u,v,x,y≤n≤1001 \leq u, v, x, y \leq n \leq 1001u,v,x,yn100,且给出的是一棵树。保证 uuuvvv 的父结点。

求深度

  • 1.构建树,用结构体更方便
  • 2.dfs对每个节点深度赋值
  • 3.找到所有节点深度最大值

求宽度

  • 1.统计每一层的宽度
  • 2.求宽度最大值

求距离
BFS

AC代码

#include<bits/stdc++.h>
using namespace std;
struct node//建树 
{int l,r,f,d;//分别代表左子节点、右子节点、父节点和当前节点深度 
}ns[105];
struct node2//便于查找距离 
{int pos,step;
};
int dp=0,wd,wid[105],st[105];//分别表示当前最大深度、宽度,宽度数组和状态数组 
void dfs(int p)
{if(st[p])return ;//如果已经被访问过return st[p]=1;//标记为1 int le=ns[p].l ,ri=ns[p].r ,de=ns[p].d;//左子节点右子节点深度赋值 dp=max(dp,de);//更新最大深度 wid[de]++;//记录宽度 if(le)//如果有左子节点 {ns[le].d=de+1;//深度加1 dfs(le);//接着向下搜 }if(ri)//如果有右子节点 {ns[ri].d =de+1;dfs(ri);}
}
int main()
{int n,u,v,x,y;cin>>n;for(int i=1;i<n;i++)//n-1条边 {cin>>u>>v;if(!ns[u].l) ns[u].l =v;//如果没有左子节点存入左子节点 else ns[u].r =v;//否则存入右子节点 ns[v].f=u;//记得更新父节点 }cin>>x>>y;//输入要查找距离的两个点 ns[1].d =1; //第一个节点即初始深度为1 dfs(1);//传入当前位置节点 for(int i=1;i<=n;i++)//循环遍历找出最大宽度 wd=max(wd,wid[i]);cout<<dp<<endl<<wd<<endl;//输出最大深度和最大宽度 memset(st,0,sizeof(st));//将状态数组重置为0 node2 tn={x,0};//将初始点和距离值传入 st[x]=1;//状态改变表示已被访问过 queue<node2>q;//建立结构体状态数组 q.push(tn);//将初始状态传进去 while(!q.empty())//当队列不空时 {tn=q.front();//取队首元素 q.pop();//队首弹出 if(tn.pos==y)//当所查找的值到达y时 {cout<<tn.step;//输出距离 break;//查找结束 }int le=ns[tn.pos].l;int ri=ns[tn.pos].r;int fa=ns[tn.pos].f;int step=tn.step ;if(le&&!st[le])//如果左子节点不空并且没被访问过 {st[le]=1;//更新状态数组 q.push({le,step+1});//将更新后的位置和距离压入队列 }if(ri&&!st[ri])//以下同理 {st[ri]=1;q.push({ri,step+1});}if(fa&&!st[fa]){st[fa]=1;q.push({fa,step+2});//因为到父节点是向上走,所以每次更新两步 }} return 0;
}

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

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

相关文章

(1)Windows环境下安装Oracle

概述&#xff1a;Oracle数据库是一种网络上的数据库, 它在网络上支持多用户, 支持服务器/客户机等部署(或配置)。服务器与客户机是软件概念&#xff1a;它们与计算机硬件不存在一一对应的关系. 即:同一台计算机既可以充当服务器又可以充当客户机,或者一台计算机只充当服务器或只…

工业数据集成中间件工具OPC Router详细介绍

一、产品概述 OPC Router 是 Software Toolbox 旗下的一款面向工业数据集成与自动化的数据中间件工具&#xff0c;专注于实现各类工业系统之间的数据交互和自动化流程编排。它通过模块化的插件机制&#xff0c;打通 PLC、ERP、MES、数据库、MQTT、REST API 等不同系统之间的数…

消息队列 2.RabbitMQ的基本概念与使用

RabbitMQ 是一款基于 AMQP&#xff08;Advanced Message Queuing Protocol&#xff09;协议的开源消息中间件&#xff0c;主要用于实现分布式系统中的消息传递&#xff0c;支持异步通信、系统解耦、流量削峰等场景。在 Java 生态中&#xff0c;RabbitMQ 被广泛应用&#xff0c;…

【web安全】SQL注入与认证绕过

目录 一、SQL注入漏洞 1.1 基础注入原理 1.2 实用注入Payload分类 逻辑绕过型 注释截断型 联合查询型 常见的万能密码-CSDN博客 二、登录绕过实战技巧 2.1 基础绕过手法 2.2 高级绕过技巧 编码绕过 多重注释 参数污染 三、密码重置漏洞利用 3.1 常见漏洞模式 3…

Python适配器模式详解:让不兼容的接口协同工作

一、模式定义与核心思想 适配器模式&#xff08;Adapter Pattern&#xff09; 是一种结构型设计模式&#xff0c;它通过创建一个中间层&#xff08;适配器&#xff09;&#xff0c;将不兼容的接口转换为客户端期望的接口。就像现实中的电源适配器&#xff0c;让不同国家的插头…

微信小程序列表数据上拉加载,下拉刷新

1.上拉加载数据&#xff0c;数据 下一页数据 前面的数据&#xff08;[...this.data.list, ...data.records&#xff09;2.当用户上拉加载过快时&#xff0c;会不停的调用接口&#xff0c;需要节流阀isLoading3.上拉加载到最后一页的判断&#xff0c;isFinish// pages/list.js…

【树上倍增 LCA DFS 前缀和】P10391 [蓝桥杯 2024 省 A] 零食采购|普及+

本文涉及知识点 C算法&#xff1a;前缀和、前缀乘积、前缀异或的原理、源码及测试用例 包括课程视频 CDFS 树上倍增 LCA P10391 [蓝桥杯 2024 省 A] 零食采购 题目描述 小蓝准备去星际旅行&#xff0c;出发前想在本星系采购一些零食&#xff0c;星系内有 nnn 颗星球&#x…

PDF发票批量打印工具哪个好?高效打印发票的实用工具推荐

开小超市这几年&#xff0c;每月要打几十张进货发票做账&#xff0c;以前打印时总犯愁&#xff1a;有的发票 PDF 太大&#xff0c;打出来字小得看不清&#xff1b;有的又太窄&#xff0c;白白浪费半张纸。试过手动调整&#xff0c;每张都要改缩放比例&#xff0c;累不说&#x…

4G模块 A7680通过MQTT协议连接到华为云

命令说明 基础AT指令 ATi显示产品的标志信息 ATCIMI查询IMSI ATCICCID从SIM卡读取ICCID ATCGSN查询产品序列号 ATCPIN查询卡状态 ATCSQ查询信号强度 ATCGATT查询当前PS域状态 ATCREG查询GPRS注册状态 ATCEREG查询4G注册状态 ATCGPADDR查询PDP地址 ATCMGF选择短信格式 ATCMGS发…

大模型词表设计与作用解析

几乎所有大型语言模型&#xff08;LLM&#xff09;都有自己独立的词表&#xff08;Vocabulary&#xff09;。这是模型设计和训练过程中的核心组件之一。以下是关于词表的关键点&#xff1a; 1. 词表的作用 分词基础&#xff1a;词表定义了模型如何将输入文本拆分成基本单元&…

(一)Eshop(异常处理中间件/grpc)

文章目录项目地址一、异常处理1.1 自定异常1.2 自定义异常处理中间件1.3 注册中间件二、grpc服务2.1 创建protos1. 打折的protos2. 设置grpc server3. program配置服务4. docker-compose2.2 CRUD1. 查询2.3 测试1. 发起查询请求三、grpc服务消费3.1 创建client1. 添加服务2. 选…

BLIP、InternVL Series(下)

目录 一、InternVL1.5 1、改进 二、InternVL2 1、渐进式扩展 2、多模态扩展 三、InternVL2.5 1、方法 2、数据优化 四、InternVL3 2、方法 3、训练后处理 4、测试时扩展 五、BLIP-3o 一、InternVL1.5 1、改进 InternVL1.5在InternVL基础上&#xff0c;优化了QLLa…

【数据结构】二维差分数组

题目链接 【模板】二维差分_牛客题霸_牛客网 牛客网 - 找工作神器|笔试题库|面试经验|实习招聘内推&#xff0c;求职就业一站解决_牛客网 描述 给定一个 nmnm 的整数矩阵 bb&#xff0c;矩阵的下标从 11 开始记作 bi,jbi,j​。现在需要支持 qq 次操作&#xff0c;第 tt 次…

【JDK内置工具】常用工具和实战指令

作者&#xff1a;唐叔在学习 专栏&#xff1a;唐叔的Java实践 关键词: #JDK工具 #Java性能调优 #JVM调优 #内存泄漏排查 #线程死锁分析 #Java开发工具 #线上问题排查 #Java诊断工具 Hello&#xff0c;大家好&#xff0c;我是爱学习的唐叔。作为Java开发者&#xff0c;JDK内置工…

一站式PDF转Markdown解决方案PDF3MD

简介 什么是 PDF3MD &#xff1f; PDF3MD 是一个现代化、用户友好的网络应用程序&#xff0c;旨在将 PDF 文档转换为干净、格式化的 Markdown 文本。它提供了高效的转换工具&#xff0c;支持多种文件格式之间的转换。 主要特点 PDF 转 Markdown&#xff1a;能够将 PDF 文档转…

RocketMQ学习系列之——MQ入门概念

一、什么是MQMQ&#xff08;Message Queue&#xff0c;消息队列&#xff09;是一种能够实现跨进程消息传输&#xff0c;并且消息缓存符合队列特性的组件。二、MQ的作用异步&#xff1a;消息发送方无需等待消息接收方收到消息&#xff0c;发送方将消息成功发送到 MQ 之后即可无阻…

血条识别功能实现及原理

从零开始学Python图像处理 - 血条识别 从实际问题中能快速的学习特定技能&#xff0c;通过完成一个能自动刷怪的工具&#xff0c;达成快速学习python图像处理和识别。 自动刷怪需要先识别怪物&#xff0c;在游戏中怪物类型很多&#xff0c;同时在移动中形态会一直发生变化&…

网络地址和主机地址之间进行转换的类

#pragma once #include "Common.hpp" // 网络地址和主机地址之间进行转换的类class InetAddr { public:InetAddr(){}InetAddr(struct sockaddr_in &addr) : _addr(addr){// 网络转主机_port ntohs(_addr.sin_port); // 从网络中拿到的&#xff01;网络序列// _i…

《Python 项目 CI/CD 实战指南:从零构建自动化部署流水线》

🛠《Python 项目 CI/CD 实战指南:从零构建自动化部署流水线》 一、引言:为什么 Python 项目需要 CI/CD? 在现代软件开发中,CI/CD(持续集成 / 持续部署)已成为不可或缺的工程实践。它不仅提升了开发效率,还显著降低了部署风险。对于 Python 项目而言,CI/CD 的价值尤…

AJAX 技术

AJAX全称是 Asynchronous JavaScript and XML ( 异步的JavaScript 和 XML )&#xff0c;使用该技术后&#xff0c;可以实现不刷新整个网页&#xff0c;与服务器进行异步通信并更新部分网页。一&#xff09;为什么需要AJAX?传统网页在与服务器通信时&#xff0c;需要刷新整个页…