文章目录
- 二叉树
- 一、二叉树理论基础篇
- 二叉树的种类
- 满二叉树
- 完全二叉树
- 二叉搜索树
- 平衡二叉搜索树
- 二叉树的存储方式
- 链式存储:
- 顺序存储:
- 遍历规则:
- 构造实现:
- 二叉树的遍历方式
- 二叉树的定义
- 二、二叉树的递归遍历
- 递归算法的三个要素:
- 递归版遍历
- 前序遍历:
- 中序遍历:
- 后序遍历:
- 迭代版遍历
- 栈:
- 前序遍历:
- 中序遍历:
- 后序遍历:
- 三、完整代码
二叉树
一、二叉树理论基础篇
二叉树的种类
满二叉树
满二叉树:如果一棵二叉树只有度为0的结点和度为2的结点,并且度为0的结点在同一层上,则这棵二叉树为满二叉树。
这棵二叉树为满二叉树,也可以说深度为k,有2^k-1
个节点的二叉树。
完全二叉树
完全二叉树:除了最底层节点可能没填满外,其余每层节点数都达到最大值,并且最下面一层的节点都集中在该层最左边的若干位置。若最底层为第 h 层,则该层包含 1~ 2^h -1
个节点。
之前我们刚刚讲过优先级队列其实是一个堆,堆就是一棵完全二叉树,同时保证父子节点的顺序关系。
二叉搜索树
前面介绍的树,都没有数值的,而二叉搜索树是有数值的了,二叉搜索树是一个有序树。
- 若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值;
- 若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值;
- 它的左、右子树也分别为二叉排序树
下面这两棵树都是搜索树:
平衡二叉搜索树
平衡二叉搜索树:又被称为AVL(Adelson-Velsky and Landis)树,且具有以下性质:它是一棵空树或它的左右两个子树的高度差的绝对值不超过1,并且左右两个子树都是一棵平衡二叉树。
C++中map、set、multimap,multiset的底层实现都是平衡二叉搜索树,所以map、set的增删操作时间时间复杂度是log(n),注意我这里没有说unordered_map、unordered_set,unordered_map、unordered_map底层实现是哈希表。
二叉树的存储方式
二叉树可以链式存储,也可以顺序存储。
链式存储:
通过指针把分布在散落在各个地址的节点串联一起。存储方式是用指针
顺序存储:
顺序存储的元素在内存是连续分布的,储的方式就是用数组
遍历规则:
用数组来存储二叉树如何遍历的呢?
如果父节点的数组下表是i,那么它的左孩子就是i \* 2 + 1
,右孩子就是 i \* 2 + 2
。
构造实现:
二叉树的遍历方式
二叉树主要有两种遍历方式:
- 深度优先遍历:先往深走,遇到叶子节点再往回走。
- 广度优先遍历:一层一层的去遍历。
-
深度优先遍历
-
- 前序遍历(递归法,迭代法)
- 中序遍历(递归法,迭代法)
- 后序遍历(递归法,迭代法)
-
广度优先遍历
-
- 层次遍历(迭代法)
这里前中后,其实指的就是中间节点的遍历顺序,只要大家记住 前中后序指的就是中间节点的位置就可以了。
- 前序遍历:中左右
- 中序遍历:左中右
- 后序遍历:左右中
最后再说一说二叉树中深度优先和广度优先遍历实现方式,我们做二叉树相关题目,经常会使用递归的方式来实现深度优先遍历,也就是实现前中后序遍历,使用递归是比较方便的。
之前我们讲栈与队列的时候,就说过栈其实就是递归的一种是实现结构,也就说前中后序遍历的逻辑其实都是可以借助栈使用非递归的方式来实现的。
而广度优先遍历的实现一般使用队列来实现,这也是队列先进先出的特点所决定的,因为需要先进先出的结构,才能一层一层的来遍历二叉树。
二叉树的定义
struct TreeNode {int val;TreeNode *left;TreeNode *right;TreeNode(int x) : val(x), left(NULL), right(NULL) {}
};
二、二叉树的递归遍历
本篇将介绍前后中序的递归写法,一些同学可能会感觉很简单,其实不然,我们要通过简单题目把方法论确定下来,有了方法论,后面才能应付复杂的递归。
递归算法的三个要素:
每次写递归,都按照这三要素来写,可以保证大家写出正确的递归算法!
- **确定递归函数的参数和返回值:**确定哪些参数是递归的过程中需要处理的,那么就在递归函数里加上这个参数, 并且还要明确每次递归的返回值是什么进而确定递归函数的返回类型。
- **确定终止条件:**写完了递归算法, 运行的时候,经常会遇到栈溢出的错误,就是没写终止条件或者终止条件写的不对,操作系统也是用一个栈的结构来保存每一层递归的信息,如果递归没有终止,操作系统的内存栈必然就会溢出。
- **确定单层递归的逻辑:**确定每一层递归需要处理的信息。在这里也就会重复调用自己来实现递归的过程。
递归版遍历
前序遍历:
class Solution {
public:void traversal(TreeNode* cur, vector<int>& vec) {if (cur == NULL) return;vec.push_back(cur->val); // 中traversal(cur->left, vec); // 左traversal(cur->right, vec); // 右}vector<int> preorderTraversal(TreeNode* root) {vector<int> result;traversal(root, result);return result;}
};
中序遍历:
void traversal(TreeNode* cur, vector<int>& vec) {if (cur == NULL) return;traversal(cur->left, vec); // 左vec.push_back(cur->val); // 中traversal(cur->right, vec); // 右
}
后序遍历:
void traversal(TreeNode* cur, vector<int>& vec) {if (cur == NULL) return;traversal(cur->left, vec); // 左traversal(cur->right, vec); // 右vec.push_back(cur->val); // 中
}
迭代版遍历
栈:
栈(Stack):是只允许在一端进行插入或删除的线性表。首先栈是一种线性表,但限定这种线性表只能在某一端进行插入和删除操作。后进先出
头文件:#include<stack>
定义:
stack<int> st;
stack<string> st;
基本操作:
s.push(x); //入栈
s.pop(); //出栈,只是删除栈顶元素,并不返回该元素
s.top(); //访问栈顶元素
s.empty(); //判断栈空,当栈空时,返回true
s.size(); //访问栈中元素个数
迭代都是依据栈Stack
来实现的
前序遍历:
前序遍历是中左右,每次先处理的是中间节点,那么先将跟节点放入栈中,然后将右孩子加入栈,再加入左孩子。
为什么要先加入 右孩子,再加入左孩子呢?
因为这样出栈的时候才是中左右的顺序。
① 进栈: 5 出栈 : 5 当前: 5
② 进栈: 64 出栈 :4 当前:6
③ 进栈: 21 出栈 : 1 当前:62
④ 进栈: 出栈 : 2 当前: 6
⑤ 进栈: 出栈 : 6 当前:
/*-----------迭代前序(左右中)-------------*/
void preTraversalByStack(TreeNode* root)
{stack<TreeNode*> st;if (root != NULL){st.push(root); //进栈while (!st.empty()){TreeNode* cur = st.top();st.pop(); //出栈printVal(cur); //中if (cur->right) st.push(cur->right); //右if (cur->left) st.push(cur->left);//左}}
}
中序遍历:
为了解释清楚,我说明一下 刚刚在迭代的过程中,其实我们有两个操作:
- 处理:将元素放进result数组中
- 访问:遍历节点
分析一下为什么刚刚写的前序遍历的代码,不能和中序遍历通用呢,因为前序遍历的顺序是中左右,先访问的元素是中间节点,要处理的元素也是中间节点,所以刚刚才能写出相对简洁的代码,因为要访问的元素和要处理的元素顺序是一致的,都是中间节点。
那么再看看中序遍历,中序遍历是左中右,先访问的是二叉树顶部的节点,然后一层一层向下访问,直到到达树左面的最底部,再开始处理节点(也就是在把节点的数值放进result数组中),这就造成了处理顺序和访问顺序是不一致的。
那么在使用迭代法写中序遍历,就需要借用指针的遍历来帮助访问节点,栈则用来处理节点上的元素。
① 进栈: 541 出栈 : 1 4 当前: 5
② 进栈: 2 出栈 :2 当前:5
③ 进栈: 出栈 : 5 当前:
④ 进栈: 6 出栈 : 6 当前:
/*-----------迭代中序(左中右)-------------*/
void midTraversalByStack(TreeNode* root)
{stack<TreeNode*> st;TreeNode* cur = root;while (cur != NULL || !st.empty()){if (cur != NULL) 指针来访问节点,访问到最底层{st.push(cur); // 将访问的节点放进栈cur = cur->left; //左}else{cur = st.top(); // 从栈里弹出的数据,就是优先遍历的数据st.pop();printVal(cur); //中cur = cur->right; // 右}}
}
后序遍历:
再来看后序遍历,先序遍历是中左右,后续遍历是左右中,那么我们只需要调整一下先序遍历的代码顺序,就变成中右左的遍历顺序,然后在反转result数组,输出的结果顺序就是左右中了,如下图:
`
/*-----------迭代后序(左右中)-------------*/
void preorderTraversalByStack(TreeNode* root)
{stack<TreeNode*> st;vector<int> result;if (root != NULL) {st.push(root);while (!st.empty()) {TreeNode* node = st.top();st.pop();result.push_back(node->val);if (node->left) st.push(node->left); // 相对于前序遍历,这更改一下入栈顺序 (空节点不入栈)if (node->right) st.push(node->right); // 空节点不入栈}reverse(result.begin(), result.end()); // 将结果反转之后就是左右中的顺序了}for (int i = 0; i< result.size(); ++i){cout << result[i] << " ";}
}
三、完整代码
#include <iostream>
#include <vector>
#include <stack>
using namespace std;struct TreeNode {int val;TreeNode *left;TreeNode *right;TreeNode(int x) : val(x), left(NULL), right(NULL) {}
};// 数组构造二叉树
TreeNode* construct_binary_tree(const vector<int>& vec) {vector<TreeNode*> vecTree(vec.size(), NULL);TreeNode* root = NULL;// 把输入数值数组,先转化为二叉树节点数组for (int i = 0; i < vec.size(); i++) {TreeNode* node = NULL;if (vec[i] != -1) node = new TreeNode(vec[i]); // 用 -1 表示nullvecTree[i] = node;if (i == 0) root = node;}// 遍历一遍,中据规则左右孩子赋值就可以了// 注意这里 结束规则是 i * 2 + 1 < vec.size(),避免空指针// 为什么结束规则不能是i * 2 + 2 < arr.length呢?// 如果i * 2 + 2 < arr.length 是结束条件// 那么i * 2 + 1这个符合条件的节点就被忽略掉了// 例如[2,7,9,-1,1,9,6,-1,-1,10] 这样的一个二叉树,最后的10就会被忽略掉// 遍历一遍,中据规则左右孩子赋值就可以了for (int i = 0; i * 2 + 1 < vec.size(); i++) {if (vecTree[i] != NULL) {// 线性存储转连式存储关键逻辑vecTree[i]->left = vecTree[i * 2 + 1];if (i * 2 + 2 < vec.size())vecTree[i]->right = vecTree[i * 2 + 2];}}return root;
}// 数组构造二叉树(简约版)
TreeNode* simple_construct_binary_tree(vector<int> array, int index)
{TreeNode* root = nullptr;if (index < array.size() && array[index] != -1){root = new TreeNode(array[index]);root->left = simple_construct_binary_tree(array, 2 * index + 1);root->right = simple_construct_binary_tree(array, 2 * index + 2);}return root;
}/*-----------打印节点的数据-------------*/
void printVal(TreeNode* cur)
{if (cur != NULL)cout << cur->val << ' ';
}/*-----------递归前序(中左右)-------------*/
void preTraversal(TreeNode* root)
{//访问跟节点if (root != NULL){printVal(root);preTraversal(root->left);preTraversal(root->right);}
}/*-----------迭代前序(左右中)-------------*/
void preTraversalByStack(TreeNode* root)
{stack<TreeNode*> st;if (root != NULL){st.push(root); //进栈while (!st.empty()){TreeNode* cur = st.top();st.pop(); //出栈printVal(cur); //中if (cur->right) st.push(cur->right); //右if (cur->left) st.push(cur->left);//左}}
}/*-----------递归中序(左中右)-------------*/
void midTraversal(TreeNode* root)
{if (root != NULL){midTraversal(root->left);printVal(root);midTraversal(root->right);}
}/*-----------迭代中序(左中右)-------------*/
void midTraversalByStack(TreeNode* root)
{stack<TreeNode*> st;TreeNode* cur = root;while (cur != NULL || !st.empty()){if (cur != NULL) 指针来访问节点,访问到最底层{st.push(cur); // 将访问的节点放进栈cur = cur->left; //左}else{cur = st.top(); // 从栈里弹出的数据,就是优先遍历的数据st.pop();printVal(cur); //中cur = cur->right; // 右}}
}/*-----------递归后序(左右中)-------------*/
void preorderTraversal(TreeNode* root)
{if (root != NULL){preorderTraversal(root->left);preorderTraversal(root->right);printVal(root);}
}/*-----------迭代后序(左右中)-------------*/
void preorderTraversalByStack(TreeNode* root)
{stack<TreeNode*> st;vector<int> result;if (root != NULL) {st.push(root);while (!st.empty()) {TreeNode* node = st.top();st.pop();result.push_back(node->val);if (node->left) st.push(node->left); // 相对于前序遍历,这更改一下入栈顺序 (空节点不入栈)if (node->right) st.push(node->right); // 空节点不入栈}reverse(result.begin(), result.end()); // 将结果反转之后就是左右中的顺序了}for (int i = 0; i< result.size(); ++i){cout << result[i] << " ";}
}int main() {vector<int> arry = { 5, 4, 6, 1, 2, -1, -1 };TreeNode* root = simple_construct_binary_tree(arry, 0);cout << "递归后序(中左右):" << endl;;preorderTraversal(root);cout << "\n迭代后序(左中右):" << endl;preorderTraversalByStack(root);system("PAUSE");return 0;
}