今天学习的是二叉树
二叉树的基础知识,
满二叉树:如果一棵二叉树只有度为0的结点和度为2的结点,并且度为0的结点在同一层上,则这棵二叉树为满二叉树
完全二叉树:在完全二叉树中,除了最底层节点可能没填满外,其余每层节点数都达到最大值,并且最下面一层的节点都集中在该层最左边的若干位置。若最底层为第 h 层,则该层包含 1~ 2^(h-1) 个节点。
二叉搜索树:
二叉搜索树是一个有序树。
- 若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值;
- 若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值;
- 它的左、右子树也分别为二叉排序树
平衡二叉搜索树:又被称为AVL(Adelson-Velsky and Landis)树,且具有以下性质:它是一棵空树或它的左右两个子树的高度差的绝对值不超过1,并且左右两个子树都是一棵平衡二叉树。
C++中map、set、multimap,multiset的底层实现都是平衡二叉搜索树,所以map、set的增删操作时间时间复杂度是logn
二叉树可以是顺序存储,即数组,也可以是链式存储,即底层采用链表实现。
二叉树遍历主要有两种
- 深度优先遍历
- 前序遍历(递归法,迭代法)
- 中序遍历(递归法,迭代法)
- 后序遍历(递归法,迭代法)
- 广度优先遍历
- 层次遍历(迭代法)
c++实现二叉树:
struct TreeNode {
int val;
TreeNode *left;
TreeNode *right;
TreeNode(int x) : val(x), left(NULL), right(NULL) {} //相比链表,树的节点有指向两个叶节点
//的指针
};
A. 使用递归算法进行遍历
递归算法需要注意的点:
-
确定递归函数的参数和返回值: 确定哪些参数是递归的过程中需要处理的,那么就在递归函数里加上这个参数, 并且还要明确每次递归的返回值是什么进而确定递归函数的返回类型。
-
确定终止条件: 写完了递归算法, 运行的时候,经常会遇到栈溢出的错误,就是没写终止条件或者终止条件写的不对,操作系统也是用一个栈的结构来保存每一层递归的信息,如果递归没有终止,操作系统的内存栈必然就会溢出。
-
确定单层递归的逻辑: 确定每一层递归需要处理的信息。在这里也就会重复调用自己来实现递归的过程。
在cur==NULL时,遍历的节点为空,即遍历结束。
/**
* Definition for a binary tree node.
* struct TreeNode {
* int val;
* TreeNode *left;
* TreeNode *right;
* TreeNode() : val(0), left(nullptr), right(nullptr) {}
* TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
* TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
* };
*/
class Solution {
public:
void Travelnode(TreeNode* cur,vector<int>& root){
if(cur == NULL) return;
root.push_back(cur->val);
Travelnode(cur->left,root);
Travelnode(cur->right,root); //中左右
}
vector<int> preorderTraversal(TreeNode* root) {
vector<int> result;
Travelnode(root,result);
return result;
}
};
class Solution {
public:
void TravelNode(TreeNode* cur, vector<int>& point){
if(cur == NULL)return;
TravelNode(cur->left ,point);
TravelNode(cur->right ,point);
point.push_back(cur->val); //左右中
}
vector<int> postorderTraversal(TreeNode* root) {
vector<int>result;
TravelNode(root,result);
return result;
}
};
题例: 94. 二叉树的中序遍历
class Solution {
public:
void TravelNode(TreeNode* cur,vector<int>& point){
if(cur == NULL)return;
TravelNode(cur->left,point);
point.push_back(cur->val);
TravelNode(cur->right,point); //左中右
}
vector<int> inorderTraversal(TreeNode* root) {
vector<int>result;
TravelNode(root,result);
return result;
}
};
B. 使用迭代法进行遍历
用栈的结构储存树的节点值,再按顺序pop出去。
前序遍历的迭代实现:
class Solution {
public:
vector<int> preorderTraversal(TreeNode* root) {
stack<TreeNode*> stack;
vector<int> result;
if(root == NULL)return result;
stack.push(root);
while(!stack.empty()){
TreeNode* tmp = stack.top();
stack.pop();
result.push_back(tmp->val);
if(tmp->right)stack.push(tmp->right);
if(tmp->left)stack.push(tmp->left);
}
return result;
}
};
但是中序和后序的场合,递归法中是先遍历到最下层节点的。而按前序的方法不能直接找到元素pop进入,这种情况需要一个指针找到最底层的元素。
class Solution {
public:
vector<int> inorderTraversal(TreeNode* root) {
stack<TreeNode*> st;
vector<int> result;
if(root == NULL)return result;
TreeNode* cur=root;
while(cur!= NULL || !st.empty()){
if(cur!=NULL){
st.push(cur);
cur=cur->left;
}
else{
cur = st.top();
st.pop();
result.push_back(cur->val);
cur=cur->right;
}
}
return result;
}
};
后续遍历的流程比较巧妙
class Solution {
public:
vector<int> postorderTraversal(TreeNode* root) {
stack<TreeNode*> st;
vector<int> result;
if(root == NULL)return result;
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());
return result;
}
};
C. 二叉树的统一迭代法
上面的三种迭代写法思路是相近的,但是在实现时考虑现实的情况导致写法上有差异。那么要如何用迭代法统一风格的写出三种的二叉树遍历呢。
关键点在于遍历的对象,确保弹出NULL之后的节点一定是想要遍历的节点。
class Solution {
public:
vector<int> inorderTraversal(TreeNode* root) {
vector<int> result;
stack<TreeNode*> st;
if (root != NULL) st.push(root);
while (!st.empty()) {
TreeNode* node = st.top();
if (node != NULL) {
st.pop(); // 将该节点弹出,避免重复操作,下面再将右中左节点添加到栈中
if (node->right) st.push(node->right); // 添加右节点(空节点不入栈)
st.push(node); // 添加中节点
st.push(NULL); // 中节点访问过,但是还没有处理,加入空节点做为标记。
if (node->left) st.push(node->left); // 添加左节点(空节点不入栈)
} else { // 只有遇到空节点的时候,才将下一个节点放进结果集
st.pop(); // 将空节点弹出
node = st.top(); // 重新取出栈中元素
st.pop();
result.push_back(node->val); // 加入到结果集
}
}
return result;
}
};
循环中将node始终赋值到st.top()上,通过调整遍历顺序来确保把正确的遍历顺序赋给node,从而只需调整左右中节点的入栈顺序即可完成不同顺序的遍历。