算法基础——二叉树

一、⼆叉树的概念

1. 二叉树的定义

注意:⼆叉树结点的两个孩⼦,⼀个被称为左孩⼦,⼀个被称为右孩⼦。其顺序是固定的,就像⼈ 的左⼿和右⼿,不能颠倒混淆。

2. 特殊的⼆叉树

(1)满⼆叉树

(2)完全⼆叉树

对⼀棵树有 个结点的⼆叉树按层序编号,所有的结点的编号从 样深度的满⼆叉树的编号为从 1 ∼n 1 ∼n 。如果这棵树所有结点和同 比特就业课 的结点位置相同,则这棵⼆叉树为完全⼆叉树。

注意:要从后往前依次删除!!!

二、⼆叉树的存储

1. 顺序存储

2. 链式存储


案例:

题⽬描述: 有⼀个 n (n ≤ 10^6 ) 个结点的⼆叉树。给出每个结点的两个⼦结点编号(均不超过n ),建⽴⼀棵⼆叉树(根节点的编号为1 ),如果是叶⼦结点,则输⼊ 0 0。

输⼊描述:
第⼀⾏⼀个整数 n ,表⽰结点数。
之后 n ⾏,第 i ⾏两个整数 lr ,分别表⽰结点 i 的左右⼦结点编号。若 l = 0 则表⽰⽆左⼦结点, r = 0 同理。

代码实现:

cpp 复制代码
#include <iostream>

using namespace std;

const int N = 1e6 + 10;

int n;
int l[N], r[N];

int main()
{   
    cin >> n;
    // 存二叉树
    for(int i = 1; i <= n; i++)
    {
        cin >> l[i] >> r[i];
    }


    return 0;
}

三、 ⼆叉树的遍历

1. 深度优先遍历

案例:

题⽬描述:
有⼀个 n (n ≤ 10^6 ) 个结点的⼆叉树。给出每个结点的两个⼦结点编号(均不超过 n ),建⽴⼀棵⼆叉树(根节点的编号为1 ),如果是叶⼦结点,则输⼊ 0 0。
输⼊描述:

第⼀⾏⼀个整数 n ,表⽰结点数。之后 n ⾏,第 i ⾏两个整数 lr ,分别表⽰结点 i 的左右⼦结点编号。若 l = 0m则表⽰⽆左⼦结点, r = 0 同理。

cpp 复制代码
测试⼀:
 4
 0 2
 3 4
 0 0
 0 0
测试⼆:
 2
 2 0
 0 0
测试三:
 3
 2 3
 0 0
 0 0
测试四:
 7
 2 3
 0 4
 5 6
 0 0
 0 0
 7 0
 0 0

代码实现:

cpp 复制代码
#include <iostream>

using namespace std;

const int N = 1e6 + 10;

int n;
int l[N], r[N]; // 存树

// 先序遍历
void dfs1(int u)
{
    cout << u << " ";
    if(l[u]) dfs1(l[u]);
    if(r[u]) dfs1(r[u]);
}

// 中序遍历
void dfs2(int u)
{
    if(l[u]) dfs2(l[u]);
    cout << u << " ";
    if(r[u]) dfs2(r[u]);
}

// 后序遍历
void dfs3(int u)
{
    if(l[u]) dfs3(l[u]);
    if(r[u]) dfs3(r[u]);
    cout << u << " ";
}

int main()
{
    cin >> n;
    for(int i = 1; i <= n; i++)
    {
        cin >> l[i] >> r[i];
    }

    dfs1(1); // 先序遍历
    cout << endl;

    dfs2(1); // 中序遍历
    cout << endl;

    dfs3(1); // 后序遍历
    cout << endl;

    return 0;
}

2. 宽度优先遍历

这个就和常规的树的遍历⽅式⼀样,直接⽤队列帮助层序遍历即可。

cpp 复制代码
#include <iostream>
#include <queue>

using namespace std;

const int N = 1e6 + 10;

int n;
int l[N], r[N];

void bfs()
{
    queue<int> q;
    q.push(1);

    while(q.size())
    {
        int u = q.front(); q.pop();
        cout << u << " ";

        if(l[u]) q.push(l[u]);
        if(r[u]) q.push(r[u]);
    }
}

int main()
{
    cin >> n;
    for(int i = 1; i <= n; i++)
    {
        cin >> l[i] >> r[i];
    }

    bfs();


    return 0;
}
相关推荐
Vacant Seat16 分钟前
贪心算法-跳跃游戏II
算法·游戏·贪心算法
夜松云25 分钟前
从对数变换到深度框架:逻辑回归与交叉熵的数学原理及PyTorch实战
pytorch·算法·逻辑回归·梯度下降·交叉熵·对数变换·sigmoid函数
杨筱毅27 分钟前
【优秀三方库研读】【性能优化点滴】odygrd/quill 解决伪共享
c++·性能优化·三方库研读
八股文领域大手子29 分钟前
深入浅出限流算法(三):追求极致精确的滑动日志
开发语言·数据结构·算法·leetcode·mybatis·哈希算法
啊阿狸不会拉杆1 小时前
人工智能数学基础(一):人工智能与数学
人工智能·python·算法
一捌年1 小时前
java排序算法-计数排序
数据结构·算法·排序算法
freexyn1 小时前
Matlab自学笔记五十二:变量名称:检查变量名称是否存在或是否与关键字冲突
人工智能·笔记·算法·matlab
YuforiaCode2 小时前
第十二届蓝桥杯 2021 C/C++组 空间
c语言·c++·蓝桥杯
YuforiaCode2 小时前
第十二届蓝桥杯 2021 C/C++组 卡片
c语言·c++·蓝桥杯
渭雨轻尘_学习计算机ing2 小时前
二叉树构建算法全解析
算法·程序员