- 1、二叉树理论基础
- 2、二叉树的递归遍历
- 3、二叉树的迭代遍历
- 4、二叉树的统一迭代法
- 5、二叉树的层序遍历
- 6、翻转二叉树
- 7、周末二叉树总结
- 8、对称二叉树
- 9、二叉树的最大深度
- 10、N叉树的最大深度
- 11、二叉树的最小深度
- 12、完全二叉树的节点个数
- 13、平衡二叉树
- 14、二叉树的所有路径
- 15、二叉树周末总结
- 16、二叉树中递归带着回溯
- 17、左叶子之和
- 18、找树左下角的值
- 19、路径总和
- 20、路径总和 II
- 21、从中序与后序遍历序列构造二叉树
- 22、从前序与中序遍历序列构造二叉树
- 23、最大二叉树
- 24、二叉树周末总结
- 25、合并二叉树
- 26、二叉树搜索树中的搜索
- 27、验证二叉搜索树
- 28、二叉搜索树的最小值绝对差
- 29、二叉搜索树中的众数
- 30、二叉树的最近公共祖先⭐
- 31、二叉树周末总结
- 32、二叉搜索树的最近公共祖先⭐
- 33、二叉搜索树中的插入操作
- 34、删除二叉搜索树中的节点
- 35、修剪二叉搜索树
- 36、将有序数组转换为二叉搜索树
- 37、把二叉树转换为累加树
- 38、二叉树总结篇

1、二叉树理论基础
1.1、二叉树的种类
我们在解题过程中,二叉树有两种主要的形式:满二叉树和完全二叉树。
1.1.1、满二叉树
满二叉树:如果一棵二叉树只有度为0的节点和度为2的节点,并且度为0的节点在同一层上,则这棵二叉树为满二叉树。
如图所示:
这棵二叉树为满二叉树,也可以说是深度为k,有 2^k - 1 个节点的二叉树。
1.1.2、完全二叉树
什么是完全二叉树?
完全二叉树的定义如下:在完全二叉树中,除了最底层的节点可能没有填满,其余每层节点数都达到最大值,并且最下面一层的节点都集中在最左边的若干位置。若最底层为第 k 层,则该层包含 1~2^(k - 1) 个节点
大家要看完全二叉树的定义,不要对完全二叉树的判断迷迷瞪瞪。
举个典型的例子:
之前我们刚刚讲过优先级队列其实是一个堆,堆就是一棵完全二叉树,同时保证父子节点的顺序关系。
1.1.3、二叉搜索树
前面介绍的树都是没有数值的,而二叉搜索数是有数值的,二叉搜索树是一棵有序树
- 若它的左子树不空,则左子树上所有节点的值均小于它的根节点的值;
- 若它的右子树不空,则右子树上所有节点的值均大于它的根节点的值;
- 它的左、右子树也分别为二叉排序树
下面这两棵树都是搜索树
1.1.4、平衡二叉搜索树
顾名思义,平衡二叉搜索树是在二叉搜索树的基础上平衡了左右子树高度。
平衡二叉搜索树:又被称为 AVL(Adelson-Velsky and Landis)数,且具有以下性质:它是一棵空树或他的左右两个子树的高度差的绝对值不超过1,并且左右两个子树都是一棵平衡二叉树。
如图:
最后一棵不是平衡二叉树,因为它的左右两个子树的高度差的绝对值超过了1。
C++中map、set、multimap、multiset 的低层实现都是平衡二叉搜索树,所以map、set的增删操作时间复杂度是 logn,注意,unordered_map 和 unordered_set 的低层实现是哈希表,增删操作时间复杂度是 O(1)。
如果对数组和链表的增删查操作的时间复杂度还有疑问的,可以看一看这篇文章
链表和数组的插入删除时间复杂度都是o(n),为什么教材网络上说链表效率高?
所以大家使用自己熟悉的编程语言写算法,一定要知道常用的容器低层都是如何实现的,最基本的就是map、set等等,否则自己写的代码,自己对其性能分析都分析不清楚!
1.2、二叉树的存储方式
二叉树可以是链式存储,也可以是顺序存储。
那么链式存储方式就用指针, 顺序存储的方式就是用数组。
顾名思义就是顺序存储的元素在内存是连续分布的,而链式存储则是通过指针把分布在散落在各个地址的节点串联一起。
链式存储如图:
链式存储是大家很熟悉的一种方式,那么我们来看看如何顺序存储呢?
其实就是用数组来存储二叉树,顺序存储的方式如图:

用数组来存储二叉树如何遍历的呢?
如果父节点的数组下标是 i,那么它的左孩子就是 i 2 + 1,右孩子就是 i 2 + 2。
但是用链式表示的二叉树,更有利于我们理解,所以一般我们都是用链式存储二叉树。
所以大家要了解,用数组依然可以表示二叉树。
1.3、二叉树的遍历方式
关于二叉树的遍历方式,要知道二叉树遍历的基本方式都有哪些。
一些同学用做了很多二叉树的题目了,可能知道前中后序遍历,可能知道层序遍历,但是却没有框架。
我这里把二叉树的几种遍历方式列出来,大家就可以一一串起来了。
二叉树主要有两种遍历方式:
- 深度优先遍历:先往深走,遇到叶子节点再往回走。
- 广度优先遍历:一层一层的去遍历。
这两种遍历是图论中最基本的两种遍历方式,后面在介绍图论的时候 还会介绍到。
那么从深度优先遍历和广度优先遍历进一步拓展,才有如下遍历方式:
- 深度优先遍历
- 前序遍历(递归法,迭代法)
- 中序遍历(递归法,迭代法)
- 后序遍历(递归法,迭代法)
- 广度优先遍历
- 层次遍历(迭代法)
在深度优先遍历中:有三个顺序,前中后序遍历, 有同学总分不清这三个顺序,经常搞混,我这里教大家一个技巧。
这里前中后,其实指的就是中间节点的遍历顺序,只要大家记住 前中后序指的就是中间节点的位置就可以了。
看如下中间节点的顺序,就可以发现,中间节点的顺序就是所谓的遍历方式
- 前序遍历:中左右
- 中序遍历:左中右
- 后序遍历:左右中
大家可以对着如下图,看看自己理解的前后中序有没有问题。
最后再说一说二叉树中深度优先和广度优先遍历实现方式,我们做二叉树相关题目,经常会使用递归的方式来实现深度优先遍历,也就是实现前中后序遍历,使用递归是比较方便的。
之前我们讲栈与队列的时候,就说过栈其实就是递归的一种是实现结构,也就说前中后序遍历的逻辑其实都是可以借助栈使用非递归的方式来实现的。
而广度优先遍历的实现一般使用队列来实现,这也是队列先进先出的特点所决定的,因为需要先进先出的结构,才能一层一层的来遍历二叉树。
这里其实我们又了解了栈与队列的一个应用场景了。
具体的实现我们后面都会讲的,这里大家先要清楚这些理论基础。
1.4、二叉树的定义
刚刚我们说过了二叉树有两种存储方式顺序存储,和链式存储,顺序存储就是用数组来存,这个定义没啥可说的,我们来看看链式存储的二叉树节点的定义方式。
struck TreeNode{int val;TreeNode* left;TreeNode* right;TreeNode(int x) : val(x), left(NULL), right(NULL){}};
大家会发现二叉树的定义 和链表是差不多的,相对于链表 ,二叉树的节点里多了一个指针, 有两个指针,指向左右孩子.
这里要提醒大家要注意二叉树节点定义的书写方式。
在现场面试的时候 面试官可能要求手写代码,所以数据结构的定义以及简单逻辑的代码一定要锻炼白纸写出来。
因为我们在刷leetcode的时候,节点的定义默认都定义好了,真到面试的时候,需要自己写节点定义的时候,有时候会一脸懵逼!
2、二叉树的递归遍历
144. 二叉树的前序遍历 - 力扣(LeetCode)
94. 二叉树的中序遍历 - 力扣(LeetCode)
145. 二叉树的后序遍历 - 力扣(LeetCode)
这里就不贴图了,题目比较简单。如果对基础的二叉树深度搜索的前中后序还有疑问的,可以看看一刷的笔记。
照例还是写递归三部曲:
- 确定递归函数的参数和返回值
- 确定终止条件
确定单层递归的逻辑
/*** 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:vector<int> preorderTraversal(TreeNode* root) {vector<int> result;preTraversal(root, result);return result;}// 前序递归void preTraversal(TreeNode* root, vector<int>& result){if(root == nullptr){return;}result.push_back(root->val); // 中preTraversal(root->left, result); // 左preTraversal(root->right, result); // 右}};
/*** 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:vector<int> inorderTraversal(TreeNode* root) {vector<int> result;midTraversal(root, result);return result;}// 中序递归void midTraversal(TreeNode* root, vector<int>& result){if(root == nullptr){return;}midTraversal(root->left, result); // 左result.push_back(root->val); // 中midTraversal(root->right, result); // 右}};
/*** 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:vector<int> postorderTraversal(TreeNode* root){vector<int> result;postTraversal(root, result);return result;}// 后序递归void postTraversal(TreeNode* root, vector<int>& result){if(root == nullptr){return;}postTraversal(root->left, result); // 左postTraversal(root->right, result); // 右result.push_back(root->val); // 中}};
3、二叉树的迭代遍历
144. 二叉树的前序遍历 - 力扣(LeetCode)
94. 二叉树的中序遍历 - 力扣(LeetCode)
145. 二叉树的后序遍历 - 力扣(LeetCode)
/*** 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:// 迭代法实现前序遍历,且是统一风格的代码vector<int> preorderTraversal(TreeNode* root) {stack<TreeNode*> st;vector<int> result;if(root == nullptr) return result;st.push(root); // 根节点先入栈while(!st.empty()){TreeNode* cur = st.top();if(cur != NULL){ // 如果节点非空,表明当前节点不是我们设置标志的节点st.pop(); // 如果当前节点非空,我们需要继续判断当前节点是否有左右孩子if(cur->right) st.push(cur->right); // 右if(cur->left) st.push(cur->left); // 左st.push(cur); // 中st.push(NULL); // 给当前节点设置标志}else{st.pop(); // 弹出空指针(即我们设置的标志)result.push_back(st.top()->val); // 这是我们想要操作的节点st.pop();}}return result;}};
/*** 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:// 迭代法:统一风格的代码vector<int> inorderTraversal(TreeNode* root) {stack<TreeNode*> st;vector<int> result;if(root == NULL) return result;st.push(root);while(!st.empty()){TreeNode* cur = st.top();if(cur != NULL){st.pop();if(cur->right) st.push(cur->right);st.push(cur);st.push(NULL);if(cur->left) st.push(cur->left);}else{st.pop();result.push_back(st.top()->val);st.pop();}}return result;}};
/*** 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: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();if(node != NULL){ // 如果节点非空,表明当前节点不是我们设置标志的节点st.pop();st.push(node); // 中st.push(NULL); // 给中节点设置标志if(node->right) st.push(node->right); // 右if(node->left) st.push(node->left); // 左}else{st.pop(); // 弹出空指针(即我们设置的标志)result.push_back(st.top()->val); // 这是我们想要操作的节点st.pop();}}return result;}};
4、二叉树的统一迭代法
在 3、二叉树的迭代遍历 我使用的就是统一的迭代法来解题的,虽然说不使用统一的解法也能把前中后序给求解出来,但是代码不同意,理解的角度也不太一样,为了方便记忆,以后就都使用统一的迭代法来解这种遍历的题目了。
5、二叉树的层序遍历
学会二叉树的层序遍历,可以一口气打完以下十题:
- 102.二叉树的层序遍历
- 107.二叉树的层次遍历II
- 199.二叉树的右视图
- 637.二叉树的层平均值
- 429.N叉树的层序遍历
- 515.在每个树行中找最大值
- 116.填充每个节点的下一个右侧节点指针
- 117.填充每个节点的下一个右侧节点指针II
- 104.二叉树的最大深度
- 111.二叉树的最小深度
5.1、二叉树的层序遍历
层序遍历一个二叉树。就是从左到右一层一层的去遍历二叉树。这种遍历的方式和我们之前讲过的都不太一样。
需要借用一个辅助数据结构即队列来实现,队列先进先出,符合一层一层遍历的逻辑,而是用栈先进后出适合模拟深度优先遍历也就是递归的逻辑。
而这种层序遍历方式就是图论中的广度优先遍历,只不过我们应用在二叉树上。
使用队列实现二叉树广度优先遍历,动画如下:
这样就实现了层序从左到右遍历二叉树。
代码如下:这份代码也可以作为二叉树层序遍历的模板,打十个就靠它了。
C++代码:
/*** 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:vector<vector<int>> levelOrder(TreeNode* root) {vector<vector<int>> result;queue<TreeNode*> que;if(root == NULL) return result;que.push(root);while(!que.empty()){vector<int> temp; // 用一个临时的一维数组来存二叉树中一层的节点int size = que.size(); // 队列的大小,即二叉树当前这一层的节点数while(size--){TreeNode* cur = que.front();temp.push_back(cur->val);que.pop();if(cur->left) que.push(cur->left); // 左节点非空就入队if(cur->right) que.push(cur->right); // 右节点非空就入队}result.push_back(temp);}return result;}};
此时我们就掌握了二叉树的层序遍历了,那么如下九道力扣上的题目,只需要修改模板的两三行代码(不能再多了),便可打倒!
5.2、二叉树的层序遍历 II
107. 二叉树的层序遍历 II - 力扣(LeetCode)

这道题目同样也是层序遍历,只不过输出的顺序是从二叉树的叶子节点所在的层到根节点所在的层,逐层从左向右遍历输出。
代码里有详细的解析
/*** 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:// 先按顺序自上向下的层序遍历,然后在最后面要输出的时候反转一下数组即可vector<vector<int>> levelOrderBottom(TreeNode* root) {vector<vector<int>> result;queue<TreeNode*> que;if(root == nullptr){return result;}que.push(root);while(!que.empty()){vector<int> temp; // 用一个临时数组来存储一层的节点元素值int size = que.size(); // 二叉树每一层的节点数while(size--){ // 遍历完一层才推出循环TreeNode* cur = que.front();temp.push_back(cur->val);que.pop();if(cur->left) que.push(cur->left);if(cur->right) que.push(cur->right);}result.push_back(temp);}reverse(result.begin(), result.end()); // 在这里反转一下数组即可return result;}};
5.3、二叉树的右视图
这道题还是层序遍历,只是要记录的是当前这一层左右边的节点而已,要实现也很简单,判断当前这一层的size等于等于最后一次循环是的值,就可以求解出来了
C++代码
/*** 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:vector<int> rightSideView(TreeNode* root) {vector<int> result;queue<TreeNode*> que;if(root == nullptr){return result;}que.push(root);while(!que.empty()){int size = que.size();while(size){TreeNode* cur = que.front();que.pop();if(size == 1){ // 如果是当前这一层中最右边的节点,那么需要把它加入到数组中result.push_back(cur->val);}if(cur->left) que.push(cur->left);if(cur->right) que.push(cur->right);size--; // 注意,因为while循环内我有size==1的判断条件,因此这里的size--不能放在while()里面}}return result;}};
5.4、二叉树的层平均值
/*** 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:vector<double> averageOfLevels(TreeNode* root) {vector<double> result;queue<TreeNode*> que;if(root == nullptr){return result;}que.push(root);while(!que.empty()){int size = que.size();int count = size; // count是一个中间变量,也就是当前层的节点数double sum = 0; // 当前这一层的和while(size--){TreeNode* cur = que.front();que.pop();sum += cur->val;if(cur->left) que.push(cur->left);if(cur->right) que.push(cur->right);}result.push_back(sum / count);}return result;}};
5.5、N 叉树的层序遍历
N叉树的层序遍历与二叉树的层序遍历简直就是一模一样的,一层之中的节点数由2个变成N个而已,没什么好讲的
C++代码
/*// Definition for a Node.class Node {public:int val;vector<Node*> children;Node() {}Node(int _val) {val = _val;}Node(int _val, vector<Node*> _children) {val = _val;children = _children;}};*/class Solution {public:vector<vector<int>> levelOrder(Node* root) {vector<vector<int>> result;queue<Node*> que;if(root == nullptr){return result;}que.push(root);while(!que.empty()){vector<int> temp; // temp用来存储N叉树当前这一层的节点int size = que.size();while(size--){Node* cur = que.front();que.pop();temp.push_back(cur->val);// 遍历N个节点for(int i = 0; i < cur->children.size(); i++){que.push(cur->children[i]);}}result.push_back(temp);}return result;}};
5.6、在每个树行中找最大值
/*** 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:vector<int> largestValues(TreeNode* root) {vector<int> result;queue<TreeNode*> que;if(root == nullptr){return result;}que.push(root);while(!que.empty()){int size = que.size();int max = INT_MIN;while(size--){TreeNode* cur = que.front();que.pop();max = std::max(max, cur->val);if(cur->left) que.push(cur->left);if(cur->right) que.push(cur->right);}result.push_back(max);}return result;}};
5.7、填充每个节点的下一个右侧节点指针
116. 填充每个节点的下一个右侧节点指针 - 力扣(LeetCode)

/*// Definition for a Node.class Node {public:int val;Node* left;Node* right;Node* next;Node() : val(0), left(NULL), right(NULL), next(NULL) {}Node(int _val) : val(_val), left(NULL), right(NULL), next(NULL) {}Node(int _val, Node* _left, Node* _right, Node* _next): val(_val), left(_left), right(_right), next(_next) {}};*/class Solution {public:Node* connect(Node* root) {if(root == NULL){return NULL;}queue<Node*> que;que.push(root);while(!que.empty()){int size = que.size();Node* pre = new Node(0); // cur的前一个节点while(size){Node* cur = que.front();que.pop();pre->next = cur; // 前一个节点的next指向当前节点if(size == 1){ // 如果是当前这一层的最右边节点,那么让它的next指向NULLcur->next = NULL;}if(cur->left) que.push(cur->left);if(cur->right) que.push(cur->right);pre = cur; // 更新pre节点size--;}}return root;}};
其实也可以不用申请 pre 节点,但是这样代码比较巧妙,有点不好想到
/*// Definition for a Node.class Node {public:int val;Node* left;Node* right;Node* next;Node() : val(0), left(NULL), right(NULL), next(NULL) {}Node(int _val) : val(_val), left(NULL), right(NULL), next(NULL) {}Node(int _val, Node* _left, Node* _right, Node* _next): val(_val), left(_left), right(_right), next(_next) {}};*/class Solution {public:Node* connect(Node* root) {if(root == NULL){return NULL;}queue<Node*> que;que.push(root);while(!que.empty()){int size = que.size();while(size){Node* cur = que.front();que.pop();if(cur->left) que.push(cur->left);if(cur->right) que.push(cur->right);cur->next = que.front(); // 更新pre节点if(size == 1){ // 如果是当前这一层的最右边节点,那么让它的next指向NULLcur->next = NULL;}size--;}}return root;}};
5.8、填充每个节点的下一个右侧节点指针 II
117. 填充每个节点的下一个右侧节点指针 II - 力扣(LeetCode)

这道题目说是二叉树,但116题目说是完整二叉树,其实没有任何差别,一样的代码一样的逻辑一样的味道
/*// Definition for a Node.class Node {public:int val;Node* left;Node* right;Node* next;Node() : val(0), left(NULL), right(NULL), next(NULL) {}Node(int _val) : val(_val), left(NULL), right(NULL), next(NULL) {}Node(int _val, Node* _left, Node* _right, Node* _next): val(_val), left(_left), right(_right), next(_next) {}};*/class Solution {public:Node* connect(Node* root) {if(root == NULL){return NULL;}queue<Node*> que;que.push(root);while(!que.empty()){int size = que.size();while(size){Node* cur = que.front();que.pop();if(cur->left) que.push(cur->left);if(cur->right) que.push(cur->right);cur->next = que.front();if(size == 1){cur->next = NULL;}size--;}}return root;}};
5.9、二叉树的最大深度
简单一点的解法,层序遍历就行了,但是效率低。也可以用递归来做,在不是极端的情况下,递归的效率略高,但也没有好到哪里去
/*** 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:int maxDepth(TreeNode* root) {if(root == nullptr){return 0;}queue<TreeNode*> que;int depth = 0;que.push(root);while(!que.empty()){int size = que.size();while(size--){TreeNode* cur = que.front();que.pop();if(cur->left) que.push(cur->left);if(cur->right) que.push(cur->right);}depth++;}return depth;}};
5.10、二叉树的最小深度
最小深度是从根节点到最近叶子节点的最短路径上的节点数量。
说明:叶子节点是指没有子节点的节点。
题目交代的很清楚,当一个节点是叶子节点的时候,它所在的路径就是最短路径
C++代码如下:
/*** 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:int minDepth(TreeNode* root) {if(root == nullptr){return 0;}queue<TreeNode*> que;int depth = 0;que.push(root);while(!que.empty()){int size = que.size();while(size--){TreeNode* cur = que.front();que.pop();if(!cur->left && !cur->right){return ++depth; // 如果是最短路径,直接return}if(cur->left) que.push(cur->left);if(cur->right) que.push(cur->right);}depth++;}return depth;}};
总结:
二叉树的层序遍历,就是图论中的广度优先搜索在二叉树中的应用,需要借助队列来实现(此时又发现队列的一个应用了)。
来吧,一口气打十个:
- 102.二叉树的层序遍历
- 107.二叉树的层次遍历II
- 199.二叉树的右视图
- 637.二叉树的层平均值
- 429.N叉树的前序遍历
- 515.在每个树行中找最大值
- 116.填充每个节点的下一个右侧节点指针
- 117.填充每个节点的下一个右侧节点指针II
- 104.二叉树的最大深度
- 111.二叉树的最小深度
致敬叶师傅!
6、翻转二叉树
6.1、递归法
要翻转整棵二叉树,思路就是翻转当前节点的左子树和右子树,然后递归这样的操作。
遍历的过程中去翻转每一个节点的左右孩子就可以达到整体翻转的效果。
注意只要把每一个节点的左右孩子翻转一下,就可以达到整体翻转的效果
这道题目使用前序遍历和后序遍历都可以,唯独中序遍历不方便,因为中序遍历会把某些节点的左右孩子翻转了两次!建议拿纸画一画,就理解了
那么层序遍历可以不可以呢?依然可以的!只要把每一个节点的左右孩子翻转一下的遍历方式都是可以的!

C++代码如下:
/*** 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:TreeNode* invertTree(TreeNode* root) {if(root == nullptr){return root;}if(root->left) invertTree(root->left); // 左if(root->right) invertTree(root->right); // 右// 中(这里代码冗余,但体现出逻辑性)if(root->left && root->right){swap(root->left, root->right);}else if(root->left && !root->right){swap(root->left, root->right);}else if(!root->left && root->right){swap(root->left, root->right);}return root;}};
递归法简化:
/*** 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:TreeNode* invertTree(TreeNode* root) {if(root == nullptr){return root;}if(root->left) invertTree(root->left); // 左if(root->right) invertTree(root->right); // 右// 中swap(root->left, root->right);return root;}};
6.2、迭代法
前序和后序都可以,都是一样的思路,稍微改一下代码就行了。我这里使用后序
/*** 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:TreeNode* invertTree(TreeNode* root) {if(root == nullptr){return root;}stack<TreeNode*> st;st.push(root);while(!st.empty()){TreeNode* cur = st.top(); // 中st.pop();swap(cur->left, cur->right);if(cur->right) st.push(cur->right); // 右if(cur->left) st.push(cur->left); // 左}return root;}};
广度优先遍历
/*** 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:TreeNode* invertTree(TreeNode* root) {queue<TreeNode*> que;if(root != nullptr){que.push(root);}while(!que.empty()){int size = que.size();for(int i = 0; i < size; i++){TreeNode* node = que.front();que.pop();swap(node->left, node->right); // 交换节点if(node->left) que.push(node->left);if(node->right) que.push(node->right);}}return root;}};
7、周末二叉树总结
8、对称二叉树
8.1、递归法**
递归三部曲
- 确定递归函数的参数和返回值
- 确定终止条件
确定单层递归的逻辑
/*** 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:bool isSymmetric(TreeNode* root) {return compare(root->left, root->right);}bool compare(TreeNode* left, TreeNode* right){// 首先排除空节点的情况if(left != nullptr && right == nullptr) return false;else if(left == nullptr && right != nullptr) return false;else if(left == nullptr && right == nullptr) return true; // 这一步很关键,后面会细讲// 排除了空节点,再排除数值不相同的情况else if(left->val != right->val) return false;// 此时就是,左右节点都存在,且数值相等的情况,这个时候才有必要做递归操作bool inside = compare(left->right, right->left); // 内测节点比较bool outsize = compare(left->left, right->right); // 外侧节点比较bool insame = inside && outsize;return insame;}};
为什么下面的代码是错误的?
/*** 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:bool isSymmetric(TreeNode* root) {return compare(root->left, root->right);}bool compare(TreeNode* left, TreeNode* right){// 首先排除空节点的情况if(left != nullptr && right == nullptr) return false;else if(left == nullptr && right != nullptr) return false;else if(left == nullptr && right == nullptr) return true;else if(left != nullptr && right != nullptr){if(left->val == right->val) return true;else return false;}// 此时就是,左右节点都存在,且数值相等的情况,这个时候才有必要做递归操作bool inside = compare(left->right, right->left);bool outsize = compare(left->left, right->right);bool insame = inside && outsize;return insame;}};
举个例子,如果测试用例是下面这样
结果输出会是 true 。为什么?
因为第一次进入方法compare,left->val == right->val,这个时候还没进入递归就return true 了,当然不对了!
所以,我们要的是,如果两个节点进行比较,并且满足 left->val == right->val,不让它们退出,还要继续进行比较,直到 它们的孩子节点都为 nullptr,我们才认为 左子树 和 右子树 是对称的。
所以这就是为什么说else if(left == nullptr && right == nullptr) return true; 这一步是很关键的。
吃透这个思路,是递归法解出这道题目的关键。
8.2、迭代法
这道题目我们也可以使用迭代法,但要注意,这里的迭代法可不是前中后序的迭代写法,因为本题的本质是判断两个树是否是相互翻转的,其实已经不是所谓二叉树遍历的前中后序的关系了。
这里我们可以使用队列来比较两个树(根节点的左右子树)是否相互翻转,(注意这不是层序遍历),虽然也是一层一层访问节点,但是和层序遍历还是有很大区别的。
使用队列
如下的条件判断和递归的逻辑是一样的。
代码如下:
/*** 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:bool isSymmetric(TreeNode* root) {queue<TreeNode*> que;que.push(root->left); // 将左子树头结点加入队列que.push(root->right); // 将右子树头结点加入队列while(!que.empty()){ // 接下来判断左子树和右子树是否对称TreeNode* leftNode = que.front();que.pop();TreeNode* rightNode = que.front();que.pop();// 左节点为空、右节点为空,此时说明是对称的,这一步要先判断,不然后面可能会对空指针进行操作if(!leftNode && !rightNode) continue;// 左右一个节点不为空,或者都不为空但数值不相同,返回falseelse if(leftNode && !rightNode) return false;else if(!leftNode && rightNode) return false;else if(leftNode->val != rightNode->val) return false;que.push(leftNode->right); // 内测节点入队que.push(rightNode->left);que.push(leftNode->left); // 外侧节点入队que.push(rightNode->right);}return true;}};
细心的话,其实可以发现,这个迭代法,其实是把左右两个子树要比较的元素顺序放进一个容器,然后成对成对的取出来进行比较,那么其实使用栈也是可以的。只是这里不再是前中后序的遍历方式。
/*** 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:bool isSymmetric(TreeNode* root) {stack<TreeNode*> st;st.push(root->left); // 将左子树头结点加入队列st.push(root->right); // 将右子树头结点加入队列while(!st.empty()){ // 接下来判断左子树和右子树是否对称TreeNode* leftNode = st.top();st.pop();TreeNode* rightNode = st.top();st.pop();// 左节点为空、右节点为空,此时说明是对称的,这一步要先判断,不然后面可能会对空指针进行操作if(!leftNode && !rightNode) continue;// 左右一个节点不为空,或者都不为空但数值不相同,返回falseelse if(leftNode && !rightNode) return false;else if(!leftNode && rightNode) return false;else if(leftNode->val != rightNode->val) return false;st.push(leftNode->right); // 内测节点入队st.push(rightNode->left);st.push(leftNode->left); // 外侧节点入队st.push(rightNode->right);}return true;}};
逻辑和使用队列是一模一样的,只是比较的顺序不一样了。队列是内测节点先比较,而栈在不改变代码逻辑的情况下,是外侧节点先比较。
总结
在迭代法中我们使用了队列,需要注意的是这不是层序遍历,而且仅仅通过一个容器来成对的存放我们要比较的元素,知道这一本质之后就发现,用队列,用栈,甚至用数组,都是可以的。
9、二叉树的最大深度
9.1、递归法
本题可以使用前序(中左右),也可以使用后序遍历(左右中),使用前序求的就是深度,使用后序求的是高度。
而根节点的高度就是二叉树的最大深度,所以本题中我们通过后序求的根节点高度来求的二叉树最大深度。
这一点其实是很多同学没有想清楚的,很多题解同样没有讲清楚。
我先用后序遍历(左右中)来计算树的高度。
后序遍历C++代码如下:
/*** 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:int getDepth(TreeNode* root){if(root == nullptr) return 0;int leftDepth = getDepth(root->left); // 左int rightDepth = getDepth(root->right); // 右int depth = 1 + max(leftDepth, rightDepth); // 中return depth;}int maxDepth(TreeNode* root) {return getDepth(root);}};
后序遍历其实就是从树的底部往上走,每往上走一层,就记录一个深度,但其实本质是求这棵树的高度。
也可以使用前序遍历来做,(充分表现出求深度回溯的过程)
/*** 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:int result = 0;void getDepth(TreeNode* root, int depth){result = result > depth ? result : depth; // 中if(root->left == nullptr && root->right == nullptr) return ;if(root->left){ // 左getDepth(root->left, depth + 1);}if(root->right){ // 右getDepth(root->right, depth + 1);}return;}int maxDepth(TreeNode* root) {if(root == nullptr) return result;getDepth(root, 1);return result;}};
可以看出使用了前序(中左右)的遍历顺序,这才是真正求深度的逻辑!
9.2、迭代法
迭代法的话使用层序遍历是最合适的,迭代法算是一道模板题了
/*** 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:int maxDepth(TreeNode* root) {if(root == nullptr) return 0;int depth = 0;queue<TreeNode*> que;que.push(root);while(!que.empty()){int size = que.size();while(size--){TreeNode* cur = que.front();que.pop();if(cur->left) que.push(cur->left);if(cur->right) que.push(cur->right);}depth++;}return depth;}};
10、N叉树的最大深度
10.1、递归法
和求解9、二叉树的最大深度的思路是一样的。
/*// Definition for a Node.class Node {public:int val;vector<Node*> children;Node() {}Node(int _val) {val = _val;}Node(int _val, vector<Node*> _children) {val = _val;children = _children;}};*/class Solution {public:int result = 0;void getDepth(Node* root, int depth){result = result > depth ? result : depth; // 中 ---这里是一定要在程序进来就处理的,不然会遗漏if(root->children.size() == 0){return;}for(int i = 0; i < root->children.size(); i++){ // 孩子节点getDepth(root->children[i], depth + 1);}return;}int maxDepth(Node* root) {if(root == NULL) return 0;getDepth(root, 1);return result;}};
代码是最好的注释,后序遍历直接看代码吧
/*// Definition for a Node.class Node {public:int val;vector<Node*> children;Node() {}Node(int _val) {val = _val;}Node(int _val, vector<Node*> _children) {val = _val;children = _children;}};*/class Solution {public:int maxDepth(Node* root) {if(root == NULL) return 0;int depth = 1;for(int i = 0; i < root->children.size(); i++){depth = max(depth, 1 + maxDepth(root->children[i]));}return depth;}};
10.2、迭代法
迭代法在之前5.5、N叉树的层序遍历的时候已经写过类似的了,不赘述
/*// Definition for a Node.class Node {public:int val;vector<Node*> children;Node() {}Node(int _val) {val = _val;}Node(int _val, vector<Node*> _children) {val = _val;children = _children;}};*/class Solution {public:int maxDepth(Node* root) {if(root == NULL) return 0;int depth = 0;queue<Node*> que;que.push(root);while(!que.empty()){int size = que.size();while(size--){Node* node = que.front();que.pop();for(int i = 0; i < node->children.size(); i++){que.push(node->children[i]);}}depth++;}return depth;}};
11、二叉树的最小深度
11.1、递归法**
前序遍历
这道题目,随想录只给出了后序遍历的做法(递归法用后序遍历确实是最理想的),于是我尝试用前序遍历来做,但是求不出来,前序遍历,每次都会遍历左子树直到底部,所以它就无法比较左右子树的深度,自然就无法求出二叉树的最小深度。
错误的代码
/*** 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:int result = 0;bool getdepth(TreeNode* root, int depth){result = result > depth ? result : depth;// 即使用标志位true和false来判断,也无法解决该题if(root->left == nullptr && root->right == nullptr){return false;}if(root->left){if(getdepth(root->left, depth + 1)){getdepth(root->left, depth + 1);}else{return false;}}if(root->right){if(getdepth(root->right, depth + 1)){getdepth(root->right, depth + 1);}else{return false;}}return true;}int minDepth(TreeNode* root) {if(root == nullptr) return 0;getdepth(root, 1);return result;}};
后序遍历
直觉上好像和求最大深度差不多,其实还是差不少的。
遍历顺序上依然是后序遍历(因为要比较递归返回之后的结果),但在处理中间节点的逻辑上,最大深度很容易理解,最小深度可有一个误区,如图:
这就重新审题了,题目中说的是:最小深度是从根节点到最近叶子节点的最短路径上的节点数量。,注意是叶子节点。
什么是叶子节点,左右孩子都为空的节点才是叶子节点!
如果这么求的话,没有左孩子的分支会算为最短深度。
所以,如果左子树为空,右子树不为空,说明最小深度是 1 + 右子树的深度。
反之,右子树为空,左子树不为空,最小深度是 1 + 左子树的深度。 最后如果左右子树都不为空,返回左右子树深度最小值 + 1 。
/*** 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:int minDepth(TreeNode* root) {if(root == nullptr) return 0;int left = minDepth(root->left); // 左int right = minDepth(root->right); // 右// 中// 如果左子树为空,右子树不为空,说明最小深度是 1 + 右子树的深度。if(root->left == nullptr && root->right != nullptr){return 1 + right;}// 如果左子树不为空,右子树为空,说明最小深度是 1 + 左子树的深度。if(root->left != nullptr && root->right == nullptr){return 1 + left;}int result = 1 + min(left, right);return result;}};
遍历的顺序为后序(左右中),可以看出:求二叉树的最小深度和求二叉树的最大深度的差别主要在于处理左右孩子不为空的逻辑。
代码精简之后
class Solution {public:int minDepth(TreeNode* root) {if (root == NULL) return 0;if (root->left == NULL && root->right != NULL) {return 1 + minDepth(root->right);}if (root->left != NULL && root->right == NULL) {return 1 + minDepth(root->left);}return 1 + min(minDepth(root->left), minDepth(root->right));}};
11.2、迭代法
迭代法还是使用 层序遍历
/*** 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:int minDepth(TreeNode* root) {if(root == nullptr) return 0;queue<TreeNode*> que;que.push(root);int result = 0;while(!que.empty()){int size = que.size();while(size--){TreeNode* cur = que.front();que.pop();if(!cur->left && !cur->right){return ++result;}if(cur->left) que.push(cur->left);if(cur->right) que.push(cur->right);}result++;}return result;}};
12、完全二叉树的节点个数
222. 完全二叉树的节点个数 - 力扣(LeetCode)

12.1、递归法
12.1.1、普通二叉树解法
/*** 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) {}* };*/// 时间复杂度:$O(\log n × \log n)$// 空间复杂度:$O(\log n)$class Solution {public:int nodes = 0;int countNodes(TreeNode* root) {if(root == nullptr) return 0;countNodes(root->left);countNodes(root->right);return 1 + nodes++; // 1 是当前节点,需要加上, 或者这样写:++nodes}};
12.1.2、完全二叉树解法
完全二叉树只有两种情况,
- 情况一:就是满二叉树
- 情况二:最后一层叶子节点没有满
对于情况一,可以直接用 2^树深度 - 1 来计算,注意,这里根节点的深度是1
对于情况二:分别递归其左孩子,右孩子,递归到某一深度一定会有左孩子或者右孩子为满二叉树,然后依然可以按照情况一来计算。
完全二叉树如图所示:
可以看出,如果一棵树不是满二叉树,就递归其左右孩子,知道遇到满二叉树为止,用公式计算这棵子树(满二叉树)的节点数量。
/*** 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:int countNodes(TreeNode* root) {if(root == nullptr) return 0;TreeNode* left = root->left;TreeNode* right = root->right;int leftHeight = 0, rightHeight = 0; // 这里初始化为0是有意的,为了下面求指数方便while(left){ // 求左子树的深度left = left->left;leftHeight++;}while(right){ // 求右子树的深度right = right->right;rightHeight++;}// 如果当前节点的左右子树的深度相同,说明是一棵满二叉树if(leftHeight == rightHeight){return(2 << leftHeight) - 1; // 注意(2<<1) 相当于2^2,所以leftHeight初始为0}// 如果当前节点所在的树不是满二叉树return countNodes(root->left) + countNodes(root->right) + 1; // 1表示当前二叉树的根节点,需要加上}};
12.2、迭代法
迭代法还是使用层序遍历是最简单的,也比较合适
/*** 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:int countNodes(TreeNode* root) {if(root == nullptr){return 0;}queue<TreeNode*> que;que.push(root);int count = 0; // 用来记录二叉树的节点个数while(!que.empty()){int size = que.size();while(size--){TreeNode* cur = que.front();que.pop();if(cur->left) que.push(cur->left);if(cur->right) que.push(cur->right);count++;}}return count;}};
13、平衡二叉树
13.1、递归法
递归三步曲分析:
- 明确递归函数的参数和返回值
参数:当前传入节点。
返回值:以当前传入节点为根节点的树的高度。
那么如何标记左右子树是否差值大于1呢?
如果当前传入节点为根节点的二叉树已经不是二叉平衡树了,还返回高度的话就没有意义了。
所以如果已经不是二叉平衡树了,可以返回-1 来标记已经不符合平衡树的规则了。
- 明确终止条件
递归的过程中依然是遇到空节点了为终止,返回0,表示当前节点为根节点的树高度为0
- 单层递归逻辑
如何判断以当前传入节点为根节点的二叉树是否是平衡二叉树呢?当然是其左子树高度和其右子树高度的差值。
分别求出其左右子树的高度,然后如果差值小于等于1,则返回当前二叉树的高度,否则则返回-1,表示已经不是二叉平衡树了。
C++代码如下:
/*** 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:bool isBalanced(TreeNode* root) {return getHeight(root) == -1 ? false : true;}// -1表示已经不是平衡二叉树了,否则返回值是以该节点为根节点树的高度int getHeight(TreeNode* root){if(root == nullptr) return 0;int leftHeight = getHeight(root->left);if(leftHeight == -1) return -1;int rightHeight = getHeight(root->right);if(rightHeight == -1) return -1;if(abs(leftHeight - rightHeight) > 1){ // 如果不是平衡二叉树,返回-1return -1;}else{return 1 + max(leftHeight, rightHeight);}}};
13.2、迭代法
在104.二叉树的最大深度(opens new window)中我们可以使用层序遍历来求深度,但是就不能直接用层序遍历来求高度了,这就体现出求高度和求深度的不同。
本题的迭代方式可以先定义一个函数,专门用来求高度。
这个函数通过栈模拟的后序遍历找每一个节点的高度(其实是通过求传入节点为根节点的最大深度来求的高度)
代码如下:
// cur节点的最大深度,就是cur的高度int getDepth(TreeNode* cur) {stack<TreeNode*> st;if (cur != NULL) st.push(cur);int depth = 0; // 记录深度int result = 0;while (!st.empty()) {TreeNode* node = st.top();if (node != NULL) {st.pop();st.push(node); // 中st.push(NULL);depth++;if (node->right) st.push(node->right); // 右if (node->left) st.push(node->left); // 左} else {st.pop();node = st.top();st.pop();depth--;}result = result > depth ? result : depth;}return result;}
然后再用栈来模拟前序遍历,遍历每一个节点的时候,再去判断左右孩子的高度是否符合,代码如下:
bool isBalanced(TreeNode* root) {stack<TreeNode*> st;if (root == NULL) return true;st.push(root);while (!st.empty()) {TreeNode* node = st.top(); // 中st.pop();if (abs(getDepth(node->left) - getDepth(node->right)) > 1) { // 判断左右孩子高度是否符合return false;}if (node->right) st.push(node->right); // 右(空节点不入栈)if (node->left) st.push(node->left); // 左(空节点不入栈)}return true;}
C++完整代码如下:
/*** 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:bool isBalanced(TreeNode* root) {stack<TreeNode*> st;if (root == NULL) return true;st.push(root);while (!st.empty()) {TreeNode* node = st.top(); // 中st.pop();if (abs(getDepth(node->left) - getDepth(node->right)) > 1) {return false;}if (node->right) st.push(node->right); // 右(空节点不入栈)if (node->left) st.push(node->left); // 左(空节点不入栈)}return true;}// cur节点的最大深度,就是cur的高度int getDepth(TreeNode* root){if(root == nullptr) return 0;stack<TreeNode*> st;st.push(root);int depth = 0; // 记录深度int result = 0;while(!st.empty()){TreeNode* node = st.top();if(node){st.pop();st.push(node);st.push(nullptr);depth++;if(node->left) st.push(node->left);if(node->right) st.push(node->right);}else{st.pop();node = st.top();st.pop();depth--;}result = result > depth ? result : depth;}return result;}};
当然此题用迭代法,其实效率很低,因为没有很好的模拟回溯的过程,所以迭代法有很多重复的计算。
虽然理论上所有的递归都可以用迭代来实现,但是有的场景难度可能比较大。
例如:都知道回溯法其实就是递归,但是很少人用迭代的方式去实现回溯算法!
因为对于回溯算法已经是非常复杂的递归了,如果在用迭代的话,就是自己给自己找麻烦,效率也并不一定高。
通过本题可以了解求二叉树深度 和 二叉树高度的差异,求深度适合用前序遍历,而求高度适合用后序遍历。
本题迭代法其实有点复杂,大家可以有一个思路,也不一定说非要写出来。
但是递归方式是一定要掌握的!
14、二叉树的所有路径
递归法:回溯
/*** 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:vector<string> result;void backTracking(TreeNode* node, string path){path += to_string(node->val); // 中if(node->left == nullptr && node->right == nullptr){result.push_back(path);return;}if(node->left) backTracking(node->left, path + "->"); // 左if(node->right) backTracking(node->right, path + "->"); // 右return;}vector<string> binaryTreePaths(TreeNode* root) {result.clear(); // 也可以不写string path;if(root == nullptr) return result;backTracking(root, path);return result;}};
迭代法
至于非递归的方式,我们可以依然可以使用前序遍历的迭代方式来模拟遍历路径的过程。
这里除了模拟递归需要一个栈,同时还需要一个栈来存放对应的遍历路径。
/*** 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:vector<string> binaryTreePaths(TreeNode* root) {vector<string> result; // 保存最终路径集合stack<string> pathSt; // 保存遍历路径的节点stack<TreeNode*> treeSt; // 保存遍历树的节点if(root == nullptr) return result;treeSt.push(root);pathSt.push(to_string(root->val));while(!treeSt.empty()){TreeNode* cur = treeSt.top(); // 获取当前节点, 中treeSt.pop();string path = pathSt.top(); // 获取当前路径pathSt.pop();if(cur->left == nullptr && cur->right == nullptr){result.push_back(path);}if(cur->right){ // 右treeSt.push(cur->right);pathSt.push(path + "->" + to_string(cur->right->val));}if(cur->left){ // 左treeSt.push(cur->left);pathSt.push(path + "->" + to_string(cur->left->val));}}return result;}};
15、二叉树周末总结
16、二叉树中递归带着回溯
由于一刷的时候我学的比较仔细,二刷基本扫一遍心里就有数了,所以这里也感觉不需要做笔记。如果有什么不懂的,可以去翻笔记,或者看卡哥的代码随想录二叉树中递归带着回溯
17、左叶子之和
递归法
因为题目中其实没有说清楚左叶子究竟是什么节点,那么我来给出左叶子的明确定义:如果左节点不为空,且左节点没有左右孩子,那么这个节点的左节点就是左叶子
/*** 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:int sum = 0;int sumOfLeftLeaves(TreeNode* root) {if(root == nullptr) return 0;// 如果当前节点是父节点的左孩子,且当前节点是叶子节点if(root->left && (!root->left->left && !root->left->right)){sum += root->left->val;}sumOfLeftLeaves(root->left);sumOfLeftLeaves(root->right);return sum;}};
卡哥的代码
class Solution {public:int sumOfLeftLeaves(TreeNode* root) {if (root == NULL) return 0;int leftValue = sumOfLeftLeaves(root->left); // 左int rightValue = sumOfLeftLeaves(root->right); // 右// 中int midValue = 0;if (root->left && !root->left->left && !root->left->right) { // 中midValue = root->left->val;}int sum = midValue + leftValue + rightValue;return sum;}};
精简之后(我还是绝的我的代码更好)
class Solution {public:int sumOfLeftLeaves(TreeNode* root) {if (root == NULL) return 0;int midValue = 0;if (root->left != NULL && root->left->left == NULL && root->left->right == NULL) {midValue = root->left->val;}return midValue + sumOfLeftLeaves(root->left) + sumOfLeftLeaves(root->right);}};
迭代法:
层序遍历也能做,所以不要被卡哥的思维框住了
/*** 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:int sumOfLeftLeaves(TreeNode* root) {queue<TreeNode*> que;que.push(root);int sum = 0;while(!que.empty()){int size = que.size();while(size--){TreeNode* cur = que.front();que.pop();if(cur->left && (!cur->left->left && !cur->left->right)){sum += cur->left->val;}if(cur->left) que.push(cur->left);if(cur->right) que.push(cur->right);}}return sum;}};
前中后序都能做,这里图方便用前序写了
/*** 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:int sumOfLeftLeaves(TreeNode* root) {stack<TreeNode*> st;st.push(root);int sum = 0;while(!st.empty()){TreeNode* cur = st.top(); // 中st.pop();if(cur->left && !cur->left->left && !cur->left->right){sum += cur->left->val;}if(cur->right) st.push(cur->right); // 左if(cur->left) st.push(cur->left); // 右}return sum;}};
18、找树左下角的值
递归法
咋眼一看,这道题目用递归的话就就一直向左遍历,最后一个就是答案呗?
没有这么简单,一直向左遍历到最后一个,它未必是最后一行啊。
我们来分析一下题目:在树的最后一行找到最左边的值。
首先要是最后一行,然后是最左边的值。
如果使用递归法,如何判断是最后一行呢,其实就是深度最大的叶子节点一定是最后一行。
如果对二叉树深度和高度还有点疑惑的话,请看:110.平衡二叉树(opens new window)。
所以要找深度最大的叶子节点。
那么如果找最左边的呢?可以使用前序遍历,这样才先优先左边搜索,然后记录深度最大的叶子节点,此时就是树的最后一行最左边的值。
递归三部曲:
- 确定递归函数的参数和返回值
参数必须有要遍历的树的根节点,还有就是一个int型的变量用来记录最长深度。 这里就不需要返回值了,所以递归函数的返回类型为void。
本题还需要类里的两个全局变量,maxLen用来记录最大深度,maxleftValue记录最大深度最左节点的数值。
int maxLen = INT_MIN; // 全局变量 记录最大深度int maxleftValue; // 全局变量 最大深度最左节点的数值void traversal(TreeNode* root, int leftLen)
有的同学可能疑惑,为啥不能递归函数的返回值返回最长深度呢?
其实很多同学都对递归函数什么时候要有返回值,什么时候不能有返回值很迷茫。
如果需要遍历整棵树,递归函数就不能有返回值。如果需要遍历某一条固定路线,递归函数就一定要有返回值!
初学者可能对这个结论不太理解,别急,后面我会安排一道题目专门讲递归函数的返回值问题。这里大家暂时先了解一下。
本题我们是要遍历整个树找到最深的叶子节点,需要遍历整棵树,所以递归函数没有返回值。
- 确定终止条件
当遇到叶子节点的时候,就需要统计一下最大的深度了,所以需要遇到叶子节点来更新最大深度。
if (root->left == NULL && root->right == NULL) {if (leftLen > maxLen) {maxLen = leftLen; // 更新最大深度maxleftValue = root->val; // 最大深度最左面的数值}return;}
- 确定单层递归的逻辑
在找最大深度的时候,递归的过程中依然要使用回溯,代码如下
if (root->left) { // 左leftLen++; // 深度加一traversal(root->left, leftLen);leftLen--; // 回溯,深度减一}if (root->right) { // 右leftLen++; // 深度加一traversal(root->right, leftLen);leftLen--; // 回溯,深度减一}return;
完整C++代码
/*** 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:int maxLen = INT_MIN; // 全局变量 记录最大深度int maxLeftValue = 0; // 全局变量 最大深度最左节点的数值void traversal(TreeNode* root, int leftLen){if(root->left == NULL && root->right == NULL){if(leftLen > maxLen){maxLen = leftLen;maxLeftValue = root->val; // 中}}if(root->left){ // 左traversal(root->left, leftLen + 1); // 回溯}if(root->right){ // 右traversal(root->right, leftLen + 1); // 回溯}}int findBottomLeftValue(TreeNode* root) {traversal(root, 1); // 根节点深度视为1return maxLeftValue;}};
迭代法:
本题使用层序遍历再合适不过了,比递归要好理解的多!
只需要记录最后一行第一个节点的数值就可以了。
/*** 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:int findBottomLeftValue(TreeNode* root) {int result = 0;queue<TreeNode*> que;que.push(root);while(!que.empty()){int size = que.size();for(int i = 0; i < size; i++){TreeNode* cur = que.front();que.pop();if(i == 0) result = cur->val;if(cur->left) que.push(cur->left);if(cur->right) que.push(cur->right);}}return result;}};
19、路径总和
递归函数什么时候需要返回值?什么时候不需要返回值?这里总结如下三点:
- 如果需要搜索整棵二叉树且不用处理递归返回值,递归函数就不要返回值。(这种情况就是本文下半部分介绍的113.路径总和ii)
- 如果需要搜索整棵二叉树且需要处理递归返回值,递归函数就需要返回值。 (这种情况我们在236. 二叉树的最近公共祖先(opens new window)中介绍)
- 如果要搜索其中一条符合条件的路径,那么递归一定需要返回值,因为遇到符合条件的路径了就要及时返回。(本题的情况)
C++完整代码
/*** 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:bool traversal(TreeNode* root, int count){if(root->left == NULL && root->right == NULL && count == 0) return true; // 遇到叶子节点,并且计数为0if(root->left == NULL && root->right == NULL) return false; // 遇到叶子节点直接返回if(root->left){ // 左// 递归函数是有返回值的,如果递归函数返回true,说明找到了合适的路径,应该立刻返回。if(traversal(root->left, count - root->left->val)){return true;}}if(root->right){ // 右// 递归函数是有返回值的,如果递归函数返回true,说明找到了合适的路径,应该立刻返回。if(traversal(root->right, count - root->right->val)){return true;}}return false;}bool hasPathSum(TreeNode* root, int targetSum){if(root == NULL) return false;return traversal(root, targetSum - root->val);}};
迭代法
如果使用栈模拟递归的话,那么如果做回溯呢?
此时栈里一个元素不仅要记录该节点指针,还要记录从头结点到该节点的路径数值总和。
c++就我们用pair结构来存放这个栈里的元素。
定义为:pair
这个为栈里的一个元素。
如下代码是使用栈模拟的前序遍历,如下:(详细注释)
/*** 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:bool hasPathSum(TreeNode* root, int targetSum){if(root == NULL) return false;stack<pair<TreeNode*, int>> st;st.push(pair<TreeNode*, int>(root, root->val));while(!st.empty()){pair<TreeNode*, int> cur = st.top();st.pop();// 如果该节点是叶子节点了,并且该节点的路径总和等于targetSum,返回trueif(cur.first->left == NULL && cur.first->right == NULL && cur.second == targetSum) return true;// 右节点,压进去一个节点的时候,将该节点的路径数值也记录下来if(cur.first->right){ // 右st.push(pair<TreeNode*, int>(cur.first->right, cur.second + cur.first->right->val));}// 左节点,压进去一个节点的时候,将该节点的路径数值也记录下来if(cur.first->left){ // 左st.push(pair<TreeNode*, int>(cur.first->left, cur.second + cur.first->left->val));}}return false;}};
20、路径总和 II
13.路径总和ii要遍历整个树,找到所有路径,所以递归函数不要返回值!
/*** 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:vector<vector<int>> result;vector<int> path;void traversal(TreeNode* node, int targetSum){if(!node->left && !node->right && targetSum == 0){result.push_back(path);}if(!node->left && !node->right) return; // 遇到叶子节点,没有合适的边直接返回if(node->left){path.push_back(node->left->val);traversal(node->left, targetSum - node->left->val);path.pop_back();}if(node->right){path.push_back(node->right->val);traversal(node->right, targetSum - node->right->val);path.pop_back();}}vector<vector<int>> pathSum(TreeNode* root, int targetSum) {if(root == nullptr) return result;path.push_back(root->val); // 头节点需要单独处理traversal(root, targetSum - root->val); // 头节点需要单独处理return result;}};
上面是一刷之后的代码,相对简洁,逻辑也清晰。如果忘了的花,去看卡哥的详细代码二叉树 - 18、路径总和 II
21、从中序与后序遍历序列构造二叉树
106. 从中序与后序遍历序列构造二叉树 - 力扣(LeetCode)

思路
首先回忆一下如何根据两个顺序构造一个唯一的二叉树,相信理论知识大家应该都清楚,就是以 后序数组的最后一个元素为切割点,先切中序数组,根据中序数组,反过来在切后序数组。一层一层切下去,每次后序数组最后一个元素就是节点元素。
为什么可以这样呢?
- 因为后序数组最后一个元素肯定就是根节点元素,我们只需要在中序数组中找到与这个根节点相同的元素,这个元素就是中序数组的根节点,
- 由中序遍历的特性可知,数组的存放形式是这样的:
[左子树,根节点,右子树],且此时左子树的长度肯定等于后序遍历中左子树的长度,即 [ 8, 9, 6] —> [ 8, 6, 9],右子树也是如此。 - 递归步骤1和2。
如果让我们肉眼看两个序列,画一棵二叉树的话,应该分分钟都可以画出来。
流程如图:
那么代码应该怎么写呢?
说到一层一层切割,就应该想到了递归。
来看一下一共分几步:
- 第一步:如果数组大小为零的话,说明是空节点了。
- 第二步:如果不为空,那么取后序数组最后一个元素作为节点元素。
- 第三步:找到后序数组最后一个元素在中序数组的位置,作为切割点
- 第四步:切割中序数组,切成中序左数组和中序右数组 (顺序别搞反了,一定是先切中序数组)
- 第五步:切割后序数组,切成后序左数组和后序右数组
- 第六步:递归处理左区间和右区间
C++完整代码:
/*** 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:TreeNode* traversal(vector<int>& inorder, vector<int>& postorder){// 第1步:如果数组大小为0的话,说明是空节点了if(inorder.size() == 0 || postorder.size() == 0) return NULL;// 第2步:如果不为空,那么取后序数组最后一个元素作为节点元素int rootVal = postorder[postorder.size() - 1];TreeNode* root = new TreeNode(rootVal);// 叶子节点if(postorder.size() == 1) return root;// 第3步:找到后序数组中最后一个元素在中序数组中的位置,作为切割点int delimiterIndex = -1;for(delimiterIndex = 0; delimiterIndex < inorder.size(); delimiterIndex++){if(inorder[delimiterIndex] == rootVal){break;}}// 第4步:切割中序数组,切成中序左数组和中序右数组// 左闭右开:[0, delimiterIndex)vector<int> leftInorder(inorder.begin(), inorder.begin() + delimiterIndex);// 左闭右开:[delimiterIndex + 1, end)vector<int> rightInorder(inorder.begin() + delimiterIndex + 1, inorder.end());// 第5步:切割后序数组,切成后序左数组和后序右数组postorder.resize(postorder.size() - 1);// 左闭右开vector<int> leftPostorder(postorder.begin(), postorder.begin() + leftInorder.size());// 左闭右开vector<int> rightPostorder(postorder.begin() + leftInorder.size(), postorder.end());// 第6步:递归处理左区间和右区间root->left = traversal(leftInorder, leftPostorder);root->right = traversal(rightInorder, rightPostorder);return root;}TreeNode* buildTree(vector<int>& inorder, vector<int>& postorder) {if (inorder.size() == 0 || postorder.size() == 0) return nullptr;return traversal(inorder, postorder);}};
此时应该发现了,如上的代码性能并不好,应为每层递归定定义了新的vector(就是数组),既耗时又耗空间,但上面的代码是最好理解的,为了方便读者理解,所以用如上的代码来讲解。
下面给出用下标索引写出的代码版本:(思路是一样的,只不过不用重复定义vector了,每次用下标索引来分割)
C++优化版本
/*** 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:// 中序区间:[inorderBegin, inorderEnd), 后序区间: [postorderBegin, postorderEnd)TreeNode* traversal(vector<int>& inorder, int inorderBegin, int inorderEnd, vector<int>& postorder, int postorderBegin, int postorderEnd){// 第1步:如果数组大小为0的话,说明是空节点了if(postorderBegin == postorderEnd) return NULL;// 第2步:如果不为空,那么取后序数组最后一个元素作为节点元素int rootVal = postorder[postorderEnd - 1];TreeNode* root = new TreeNode(rootVal);// 叶子节点if(postorderEnd - postorderBegin == 1) return root;// 第3步:找到后序数组中最后一个元素在中序数组中的位置,作为切割点int delimiterIndex = 0;for(delimiterIndex = 0; delimiterIndex < inorderEnd; delimiterIndex++){if(inorder[delimiterIndex] == rootVal){break;}}// 第4步:切割中序数组,切成中序左数组和中序右数组// 左中序区间: 左闭右开 [leftInorderBegin, leftInorderEnd)int leftInorderBegin = inorderBegin;int leftInorderEnd = delimiterIndex;// 右中序区间: 左闭右开 [rightInorderBegin, rightInorderEnd)int rightInorderBegin = delimiterIndex + 1;int rightInorderEnd = inorderEnd;// 第5步:切割后序数组,切成后序左数组和后序右数组// 后序左数组: 左闭右开 [leftPostorderBegin, leftPostorderEnd)int leftPostorderBegin = postorderBegin;int leftPostorderEnd = postorderBegin + leftInorderEnd - leftInorderBegin;// 后序右数组: 左闭右开 [rightPostorderBegin, rightPostorderEnd)int rightPostorderBegin = leftPostorderEnd;int rightPostorderEnd = postorderEnd - 1;// 第6步:递归处理左区间和右区间root->left = traversal(inorder, leftInorderBegin, leftInorderEnd, postorder, leftPostorderBegin, leftPostorderEnd);root->right = traversal(inorder, rightInorderBegin, rightInorderEnd, postorder, rightPostorderBegin, rightPostorderEnd);return root;}TreeNode* buildTree(vector<int>& inorder, vector<int>& postorder) {if (inorder.size() == 0 || postorder.size() == 0) return nullptr;return traversal(inorder, 0, inorder.size(), postorder, 0, postorder.size());}};
22、从前序与中序遍历序列构造二叉树
105. 从前序与中序遍历序列构造二叉树 - 力扣(LeetCode)

跟106.从中序与后序遍历序列构造二叉树思路是雷同的,只是有些细节需要改动,具体见代码
C++完整代码
/*** 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:TreeNode* traversal(vector<int>& preorder, int PreBegin, int PreEnd, vector<int>& inorder, int inBegin, int inEnd){// 第1步:如果数组大小为0,说明是空节点了if(PreEnd - PreBegin == 0) return nullptr;// 第2步:如果不为空,那么取前序数组的第一个元素作为节点元素int rootVal = preorder[PreBegin];TreeNode* root = new TreeNode(rootVal);// 如果是叶子节点if(PreEnd - PreBegin == 1) return root;// 第3步:找到前序数组第一个元素在中序数组中的位置int delimiterIndex = 0;for(delimiterIndex = 0; delimiterIndex < inEnd; delimiterIndex++){if(inorder[delimiterIndex] == rootVal) break;}// 第4步:分割中序数组// 左中序数组int leftInBegin = inBegin;int leftInEnd = delimiterIndex;// 右中序数组int rightInBegin = leftInEnd + 1;int rightInEnd = inEnd;// 第5步:切割前序数组// 左前序数组:左闭右开->[leftPreBegin, leftPreEnd)int leftPreBegin = PreBegin + 1;int leftPreEnd = PreBegin + 1 + leftInEnd - leftInBegin;// 右前序数组:左闭右开->[rightPreBegin, rightPreEnd)int rightPreBegin = leftPreEnd;int rightPreEnd = PreEnd;// 第6步:递归操作root->left = traversal(preorder, leftPreBegin, leftPreEnd, inorder, leftInBegin, leftInEnd);root->right = traversal(preorder, rightPreBegin, rightPreEnd, inorder, rightInBegin, rightInEnd);return root;}TreeNode* buildTree(vector<int>& preorder, vector<int>& inorder) {if(preorder.size() == 0 || inorder.size() == 0) return nullptr;return traversal(preorder, 0, preorder.size(), inorder, 0, inorder.size());}};
23、最大二叉树
这道题和 从中序与后序遍历序列构造二叉树 如出一辙,甚至比它更加简单,因为这道题不用 递归分割两个数组,只需要考虑分割一个数组的情况就行了。啥都不说,直接上代码。
/*** 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:TreeNode* traversal(vector<int>& nums){// 第1步:如果数组大小为0,说明是空节点了if(nums.size() == 0) return nullptr;// 第2步:获得nums中的最大值int delimiterIndex = 0; // 记录数组nums中最大值的索引int rootVal = INT_MIN;for(int i = 0; i < nums.size(); i++){if(nums[i] > rootVal){rootVal = nums[i];delimiterIndex = i;}}// 第3步:创建一个根节点TreeNode* root = new TreeNode(rootVal);if(nums.end() - nums.begin() == 1) return root; // 如果是叶子节点,直接返回// 第4步:分割左右两个数组, 左闭右开区间vector<int> leftNums(nums.begin(), nums.begin() + delimiterIndex);vector<int> rightNums(nums.begin() + delimiterIndex + 1, nums.end());// 第5步:递归操作root->left = traversal(leftNums);root->right = traversal(rightNums);return root;}TreeNode* constructMaximumBinaryTree(vector<int>& nums) {return traversal(nums);}};
跟中后序数组构造二叉树一样,这道题也可以对代码进行优化,优化的部分也是针对在递归操作中反复创建的vector对象。
/*** 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:TreeNode* traversal(vector<int>& nums, int begin, int end){// 第1步:如果数组大小为0,说明是空节点了if(end - begin <= 0) return nullptr;// 第2步:获得nums中的最大值int delimiterIndex = begin; // 记录数组nums中最大值的索引for(int i = begin + 1; i < end; i++){if(nums[i] > nums[delimiterIndex]){delimiterIndex = i;}}// 第3步:创建一个根节点TreeNode* root = new TreeNode(nums[delimiterIndex]);if(end - begin == 1) return root; // 如果是叶子节点,直接返回// 第4步:分割左右两个数组, 左闭右开区间int leftbegin = begin;int leftend = delimiterIndex;int rightbegin = delimiterIndex + 1;int rightend = end;// 第5步:递归操作root->left = traversal(nums, leftbegin, leftend);root->right = traversal(nums, rightbegin, rightend);return root;}TreeNode* constructMaximumBinaryTree(vector<int>& nums) {return traversal(nums, 0, nums.size());}};
注意类似用数组构造二叉树的题目,每次分隔尽量不要定义新的数组,而是通过下标索引直接在原数组上操作,这样可以节约时间和空间上的开销。
一些同学也会疑惑,什么时候递归函数前面加if,什么时候不加if,这个问题我在最后也给出了解释。
其实就是不同代码风格的实现,一般情况来说:如果让空节点(空指针)进入递归,就不加if,如果不让空节点进入递归,就加if限制一下, 终止条件也会相应的调整。
24、二叉树周末总结
25、合并二叉树
相信这道题目很多同学疑惑的点是如何同时遍历两个二叉树呢?
其实和遍历一个树逻辑是一样的,只不过传入两个树的节点,同时操作。
递归
二叉树使用递归,就要想使用前中后哪种遍历方式?
本题使用哪种遍历都是可以的!
我们下面以前序遍历为例。
/*** 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:TreeNode* combine(TreeNode* root1, TreeNode* root2){// 递归出口if(root1 == nullptr) return root2; // 如果root1为空,合并之后就应该是root2if(root2 == nullptr) return root1; // 如果root2为空,合并之后就应该是root1// 单层递归逻辑if(root1 != nullptr && root2 != nullptr){ // 中root1->val = root1->val + root2->val;}root1->left = combine(root1->left, root2->left); // 左root1->right = combine(root1->right, root2->right); // 右return root1;}TreeNode* mergeTrees(TreeNode* root1, TreeNode* root2) {if(root1 == nullptr && root2 == nullptr) return nullptr;return combine(root1, root2);}};
中序后序也可以,改一下代码顺序而已。但是前序遍历是最好理解的,我建议大家用前序遍历来做就OK。
如上的方法修改了t1的结构,当然也可以不修改t1和t2的结构,重新定一个树。
不修改输入树的结构,前序遍历,代码如下:
class Solution {public:TreeNode* mergeTrees(TreeNode* t1, TreeNode* t2) {if (t1 == NULL) return t2;if (t2 == NULL) return t1;// 重新定义新的节点,不修改原有两个树的结构TreeNode* root = new TreeNode(0);root->val = t1->val + t2->val;root->left = mergeTrees(t1->left, t2->left);root->right = mergeTrees(t1->right, t2->right);return root;}};
迭代法
/*** 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:TreeNode* mergeTrees(TreeNode* root1, TreeNode* root2) {if(root1 == nullptr) return root2;if(root2 == nullptr) return root1;queue<TreeNode*> que;que.push(root1);que.push(root2);while(!que.empty()){TreeNode* node1 = que.front(); que.pop();TreeNode* node2 = que.front(); que.pop();// 后面的程序对非空节点不能入队,所以这里node1和node2肯定不为空node1->val = node1->val + node2->val;// 如果node1->left和node2->left都不为空,那么一起入队if(node1->left && node2->left){que.push(node1->left);que.push(node2->left);}// 如果node1->right和node2->right都不为空,那么一起入队if(node1->right && node2->right){que.push(node1->right);que.push(node2->right);}// 如果node1->left为空,node2->left不为空,将node2->right赋给node1->leftif(!node1->left && node2->left){node1->left = node2->left;}// 如果node1->right为空,node2->right不为空,将node2->right赋给node1->rightif(!node1->right && node2->right){node1->right = node2->right;}}return root1;}};
合并二叉树,也是二叉树操作的经典题目,如果没有接触过的话,其实并不简单,因为我们习惯了操作一个二叉树,一起操作两个二叉树,还会有点懵懵的。
这不是我们第一次操作两棵二叉树了,在二叉树:我对称么?(opens new window)中也一起操作了两棵二叉树。
迭代法中,一般一起操作两个树都是使用队列模拟类似层序遍历,同时处理两个树的节点,这种方式最好理解,如果用模拟递归的思路的话,要复杂一些。
26、二叉树搜索树中的搜索
递归法
二叉搜索树是一个有序树:
- 若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值;
- 若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值;
- 它的左、右子树也分别为二叉搜索树
这就决定了,二叉搜索树,递归遍历和迭代遍历和普通二叉树都不一样。
- 确定递归函数的参数以及返回值
- 确定终止条件 如果root为空,或者找到这个数值了,就返回root节点。
- 确定单层递归逻辑 如果找到了如何条件的节点,return 即可
使用二叉搜索树解出来之后,尝试了以下用普通二叉树的解法来解,发现在保证代码简洁的情况下似乎不能解出来(当然,如果硬解肯定是可以的,但是如果变复杂了就觉得没有必要)。/*** 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:TreeNode* searchBST(TreeNode* root, int val) {if(root == nullptr || root->val == val) return root;if(root->val < val) return searchBST(root->right, val);else if(root->val > val) return searchBST(root->left, val);return nullptr;}};
但事实上,我是傻逼,自己功力不够而已,二叉树递归法代码也相当简洁/*** 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:TreeNode* searchBST(TreeNode* root, int val) {if(root == NULL || root->val == val) return root;TreeNode* left = searchBST(root->left, val);if(left != NULL){return left;}return searchBST(root->right, val);}};
迭代法
/*** 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:TreeNode* searchBST(TreeNode* root, int val) {while(root != NULL){if(root->val > val) root = root->left;else if(root->val < val) root = root->right;else if(root->val == val) return root;}return NULL;}};
代码简单的让人痛哭流涕。
27、验证二叉搜索树
要知道中序遍历下,输出的二叉搜索树节点的数值是有序序列。
有了这个特性,验证二叉搜索树,就相当于变成了判断一个序列是不是递增的了。
27.1、递归法
可以递归中序遍历将二叉搜索树转变成一个数组,然后只要比较一下,这个数组是否是有序的,注意二叉搜索树中不能有重复元素。
/*** 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:vector<int> vec;void traversal(TreeNode* root){if(root == nullptr) return;traversal(root->left);vec.push_back(root->val); // 将二叉搜索树转换为有序数组traversal(root->right);}bool isValidBST(TreeNode* root) {vec.clear();traversal(root);for(int i = 1; i < vec.size(); i++){if(vec[i] <= vec[i - 1]) return false;}return true;}};
以上代码中,我们把二叉树转变为数组来判断,是最直观的,但其实不用转变成数组,可以在递归遍历的过程中直接判断是否有序。
这道题目比较容易陷入两个陷阱:
- 陷阱1
不能单纯的比较左节点小于中间节点,右节点大于中间节点就完事了。
写出了类似这样的代码:
if (root->val > root->left->val && root->val < root->right->val) {return true;} else {return false;}
我二刷的时候居然还是写出这样的代码!真是绝望!
我们要比较的是 左子树所有节点小于中间节点,右子树所有节点大于中间节点。所以以上代码的判断逻辑是错误的。
例如: [10,5,15,null,null,6,20] 这个case:
- 陷阱2
样例中最小节点 可能是int的最小值,如果这样使用最小的int来比较也是不行的。
此时可以初始化比较元素为longlong的最小值。
问题可以进一步演进:如果样例中根节点的val 可能是longlong的最小值 又要怎么办呢?文中会解答。
递归三部曲:
- 确定递归函数,返回值以及参数
要定义一个longlong的全局变量,用来比较遍历的节点是否有序,因为后台测试数据中有int最小值,所以定义为longlong的类型,初始化为longlong最小值。
注意递归函数要有bool类型的返回值, 我们在二叉树:递归函数究竟什么时候需要返回值,什么时候不要返回值?(opens new window)中讲了,只有寻找某一条边(或者一个节点)的时候,递归函数会有bool类型的返回值。
其实本题是同样的道理,我们在寻找一个不符合条件的节点,如果没有找到这个节点就遍历了整个树,如果找到不符合的节点了,立刻返回。
long long maxVal = LONG_MIN; // 因为后台测试数据中有int最小值bool isValidBST(TreeNode* root)
- 确定终止条件
如果是空节点 是不是二叉搜索树呢?
是的,二叉搜索树也可以为空!
if (root == NULL) return true;
- 确定单层递归的逻辑
中序遍历,一直更新maxVal,一旦发现maxVal >= root->val,就返回false,注意元素相同时候也要返回false。
bool left = isValidBST(root->left); // 左// 中序遍历,验证遍历的元素是不是从小到大if (maxVal < root->val) maxVal = root->val; // 中else return false;bool right = isValidBST(root->right); // 右return left && right;
整体代码如下:
class Solution {public:long long maxVal = LONG_MIN; // 因为后台测试数据中有int最小值bool isValidBST(TreeNode* root) {if (root == NULL) return true;bool left = isValidBST(root->left);// 中序遍历,验证遍历的元素是不是从小到大if (maxVal < root->val) maxVal = root->val;else return false;bool right = isValidBST(root->right);return left && right;}};
以上代码是因为后台数据有int最小值测试用例,所以都把maxVal改成了longlong最小值。
如果测试数据中有 longlong的最小值,怎么办?
不可能在初始化一个更小的值了吧。 建议避免 初始化最小值,如下方法取到最左面节点的数值来比较。
/*** 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:TreeNode* pre = nullptr; // 用来记录前一个节点bool isValidBST(TreeNode* root) {if(root == nullptr) return true; // 空树也是一棵二叉搜索树bool left = isValidBST(root->left); // 左// 中序遍历,验证遍历的元素是不是从小到大if(pre && pre->val >= root->val) return false; // 中pre = root;bool right = isValidBST(root->right); // 右return left && right;}};
27.2、迭代法
可以用迭代法模拟二叉树中序遍历,对前中后序迭代法生疏的同学可以看这两篇二叉树:听说递归能做的,栈也能做!(opens new window),二叉树:前中后序迭代方式统一写法(opens new window)
迭代法中序遍历稍加改动就可以了,代码如下:
class Solution {public:bool isValidBST(TreeNode* root) {stack<TreeNode*> st;TreeNode* cur = root;TreeNode* pre = NULL; // 记录前一个节点while (cur != NULL || !st.empty()) {if (cur != NULL) {st.push(cur);cur = cur->left; // 左} else {cur = st.top(); // 中st.pop();if (pre != NULL && cur->val <= pre->val)return false;pre = cur; //保存前一个访问的结点cur = cur->right; // 右}}return true;}};
卡哥用的是非统一风格的中序遍历代码,下面我给出统一风格代码的中序遍历解法
/*** 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:bool isValidBST(TreeNode* root) {stack<TreeNode*> st;st.push(root);TreeNode* pre = nullptr;while(!st.empty()){TreeNode* cur = st.top();if(cur != nullptr){st.pop();if(cur->right) st.push(cur->right); // 右st.push(cur); // 中st.push(nullptr);if(cur->left) st.push(cur->left); // 左}else{st.pop(); // 弹出标记的节点 nullptrif(pre != nullptr && pre->val >= st.top()->val){return false;}pre = st.top(); // 记录前一个节点st.pop(); // 弹出当前节点}}return true;}};
28、二叉搜索树的最小值绝对差
530. 二叉搜索树的最小绝对差 - 力扣(LeetCode)

这道题的思路和验证二叉搜索树相似,同样不能只是比较父节点和它的左右孩子节点的差,因为有可能父节点与孙子节点的差值更小,在验证二叉搜索树中也有相同的解法,不懂往上翻,话不多说,直接上代码。
二叉搜索树还得是中序遍历
/*** 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:vector<int> vec;void traversal(TreeNode* root){if(root == nullptr) return;traversal(root->left);vec.push_back(root->val);traversal(root->right);}int getMinimumDifference(TreeNode* root) {vec.clear();traversal(root);int result = INT_MAX;for(int i = 1; i < vec.size(); i++){if(vec[i] - vec[i - 1] < result){result = vec[i] - vec[i - 1];}}return result;}};
使用pre指针
/*** 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:int result = INT_MAX;TreeNode* pre = nullptr;void traversal(TreeNode* root){if(root == nullptr) return;traversal(root->left);if(pre != nullptr){result = min(result, abs(pre->val - root->val));}pre = root; // 跟新pre指针,这一步比较难想到traversal(root->right);return;}int getMinimumDifference(TreeNode* root) {traversal(root);return result;}};
迭代法
迭代法的思路也是和验证二叉搜索树一样的
/*** 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:int getMinimumDifference(TreeNode* root) {stack<TreeNode*> st;st.push(root);int result = INT_MAX;TreeNode* pre = nullptr;while(!st.empty()){TreeNode* cur = st.top();if(cur != nullptr){st.pop();if(cur->right) st.push(cur->right);st.push(cur);st.push(nullptr); // 设置标志位if(cur->left) st.push(cur->left);}else{st.pop(); // 弹出nullptrif(pre != nullptr){result = min(result, abs(pre->val - st.top()->val));}pre = st.top(); // 更新pre指针st.pop();}}return result;}};
29、二叉搜索树中的众数
29.1、递归法
如果不是二叉搜索树
如果不是二叉搜索树,最直观的方法一定是把这个树都遍历了,用map统计频率,把频率排个序,最后取前面高频的元素的集合。
具体步骤如下:
- 这个树都遍历了,用map统计频率
至于用前中后序那种遍历也不重要,因为就是要全遍历一遍,怎么个遍历法都行,层序遍历都没毛病!
这里采用前序遍历,代码如下:
// map<int, int> key:元素,value:出现频率void searchBST(TreeNode* cur, unordered_map<int, int>& map) { // 前序遍历if (cur == NULL) return ;map[cur->val]++; // 统计元素频率searchBST(cur->left, map);searchBST(cur->right, map);return ;}
- 把统计的出来的出现频率(即map中的value)排个序
有的同学可能可以想直接对map中的value排序,还真做不到,C++中如果使用std::map或者std::multimap可以对key排序,但不能对value排序。
所以要把map转化数组即vector,再进行排序,当然vector里面放的也是pair
代码如下:
bool static cmp (const pair<int, int>& a, const pair<int, int>& b) {return a.second > b.second; // 按照频率从大到小排序}vector<pair<int, int>> vec(map.begin(), map.end());sort(vec.begin(), vec.end(), cmp); // 给频率排个序
- 取前面高频的元素
此时数组vector中已经是存放着按照频率排好序的pair,那么把前面高频的元素取出来就可以了。
代码如下
result.push_back(vec[0].first);for (int i = 1; i < vec.size(); i++) {// 取最高的放到result数组中if (vec[i].second == vec[0].second) result.push_back(vec[i].first);else break;}return result;
整体C++代码如下:
class Solution {public:// 前序遍历void searchBST(TreeNode* root, unordered_map<int, int>& map){if(root == nullptr) return;map[root->val]++; // 中searchBST(root->left, map); // 左searchBST(root->right, map); // 中}// 自定义sort比较规则, 按照频率大小从大到小排序bool static cmp(const pair<int, int>& a, const pair<int, int>& b){return a.second > b.second;}vector<int> findMode(TreeNode* root) {unordered_map<int, int> map;vector<int> result;if(root == nullptr) return result;searchBST(root, map);vector<pair<int, int>> vec(map.begin(), map.end());// 对map进行排序sort(vec.begin(), vec.end(), cmp);for(int i = 0; i < vec.size(); i++){if(vec[i].second == vec[0].second){result.push_back(vec[i].first);}else break;}return result;}};
所以如果本题没有说是二叉搜索树的话,那么就按照上面的思路写!
是二叉搜索树
既然是搜索树,它中序遍历就是有序的。
中序遍历代码如下:
void searchBST(TreeNode* cur) {if (cur == NULL) return ;searchBST(cur->left); // 左(处理节点) // 中searchBST(cur->right); // 右return ;}
遍历有序数组的元素出现频率,从头遍历,那么一定是相邻两个元素作比较,然后就把出现频率最高的元素输出就可以了。
关键是在有序数组上的话,好搞,在树上怎么搞呢?
这就考察对树的操作了。
在二叉树:搜索树的最小绝对差(opens new window)中我们就使用了pre指针和cur指针的技巧,这次又用上了。
弄一个指针指向前一个节点,这样每次cur(当前节点)才能和pre(前一个节点)作比较。
而且初始化的时候pre = NULL,这样当pre为NULL时候,我们就知道这是比较的第一个元素。
代码如下:
if (pre == NULL) { // 第一个节点count = 1; // 频率为1} else if (pre->val == cur->val) { // 与前一个节点数值相同count++;} else { // 与前一个节点数值不同count = 1;}pre = cur; // 更新上一个节点
此时又有问题了,因为要求最大频率的元素集合(注意是集合,不是一个元素,可以有多个众数),如果是数组上大家一般怎么办?
应该是先遍历一遍数组,找出最大频率(maxCount),然后再重新遍历一遍数组把出现频率为maxCount的元素放进集合。(因为众数有多个)
这种方式遍历了两遍数组。
那么我们遍历两遍二叉搜索树,把众数集合算出来也是可以的。
但这里其实只需要遍历一次就可以找到所有的众数。
那么如何只遍历一遍呢?
如果 频率count 等于 maxCount(最大频率),当然要把这个元素加入到结果集中(以下代码为result数组),代码如下:
if (count == maxCount) { // 如果和最大值相同,放进result中result.push_back(cur->val);}
是不是感觉这里有问题,result怎么能轻易就把元素放进去了呢,万一,这个maxCount此时还不是真正最大频率呢。
所以下面要做如下操作:
频率count 大于 maxCount的时候,不仅要更新maxCount,而且要清空结果集(以下代码为result数组),因为结果集之前的元素都失效了。
if (count > maxCount) { // 如果计数大于最大值maxCount = count; // 更新最大频率result.clear(); // 很关键的一步,不要忘记清空result,之前result里的元素都失效了result.push_back(cur->val);}
关键代码都讲完了,完整代码如下:(只需要遍历一遍二叉搜索树,就求出了众数的集合)
lass Solution {private:int maxCount; // 最大频率int count; // 统计频率TreeNode* pre;vector<int> result;void searchBST(TreeNode* cur) {if (cur == NULL) return ;searchBST(cur->left); // 左// 中if (pre == NULL) { // 第一个节点count = 1;} else if (pre->val == cur->val) { // 与前一个节点数值相同count++;} else { // 与前一个节点数值不同count = 1;}pre = cur; // 更新上一个节点if (count == maxCount) { // 如果和最大值相同,放进result中result.push_back(cur->val);}if (count > maxCount) { // 如果计数大于最大值频率maxCount = count; // 更新最大频率result.clear(); // 很关键的一步,不要忘记清空result,之前result里的元素都失效了result.push_back(cur->val);}searchBST(cur->right); // 右return ;}public:vector<int> findMode(TreeNode* root) {count = 0;maxCount = 0;TreeNode* pre = NULL; // 记录前一个节点result.clear();searchBST(root);return result;}};
29.2、迭代法
只要把中序遍历转成迭代,中间节点的处理逻辑完全一样。
二叉树前中后序转迭代,传送门:
下面我给出其中的一种中序遍历的迭代法,其中间处理逻辑一点都没有变(我从递归法直接粘过来的代码,连注释都没改,哈哈)
这里我使用统一风格的代码的中序遍历
/*** 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 {private:int maxCount;int count;vector<int> result;TreeNode* pre;public:void searchBST(TreeNode* cur){if(cur == nullptr) return;searchBST(cur->left); // 左// 中if(pre == nullptr){ // 第一个节点count = 1;}if(pre && pre->val == cur->val){ // 与前一个节点数值相同count++;}else{ // 与前一个节点数值不同count = 1;}pre = cur;if(count == maxCount){ // 如果和最大频率相同,那么记录它result.push_back(cur->val);}if(count > maxCount){ // 如果计数大于最大频率maxCount = count; // 更新最大频率result.clear(); // 需要对之前记录的结果清除,很关键的一步result.push_back(cur->val); // 记录新的出现频率最高的元素}searchBST(cur->right); // 右}vector<int> findMode(TreeNode* root) {maxCount = 0;count = 0;result.clear();pre = nullptr;searchBST(root);return result;}};
30、二叉树的最近公共祖先⭐
236. 二叉树的最近公共祖先 - 力扣(LeetCode)

遇到这个题目首先想的是要是能自底向上查找就好了,这样就可以找到公共祖先了。
那么二叉树如何可以自底向上查找呢?
回溯啊,二叉树回溯的过程就是从低到上。
后序遍历就是天然的回溯过程,最先处理的一定是叶子节点。
接下来就看如何判断一个节点是节点q和节点p的公共公共祖先呢。
首先最容易想到的一个情况:如果找到一个节点,发现左子树出现结点p,右子树出现节点q,或者 左子树出现结点q,右子树出现节点p,那么该节点就是节点p和q的最近公共祖先。
但是很多人容易忽略一个情况,就是节点本身p(q),它拥有一个子孙节点q(p)。
使用后序遍历,回溯的过程,就是从低向上遍历节点,一旦发现满足第一种情况的节点,就是最近公共节点了。
但是如果p或者q本身就是最近公共祖先呢?其实只需要找到一个节点是p或者q的时候,直接返回当前节点,无需继续递归子树。如果接下来的遍历中找到了后继节点满足第一种情况则修改返回值为后继节点,否则,继续返回已找到的节点即可。为什么满足第一种情况的节点一定是p或q的后继节点呢?大家可以仔细思考一下。
递归三部曲:
- 确定递归函数返回值以及参数
需要递归函数返回值,来告诉我们是否找到节点q或者p,那么返回值为bool类型就可以了。
但我们还要返回最近公共节点,可以利用上题目中返回值是TreeNode * ,那么如果遇到p或者q,就把q或者p返回,返回值不为空,就说明找到了q或者p。
TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q)
- 确定终止条件
如果找到了 节点p或者q,或者遇到空节点,就返回。
if (root == q || root == p || root == NULL) return root;
- 确定单层递归逻辑
值得注意的是 本题函数有返回值,是因为回溯的过程需要递归函数的返回值做判断,但本题我们依然要遍历树的所有节点。
我们在二叉树:递归函数究竟什么时候需要返回值,什么时候不要返回值?(opens new window)中说了 递归函数有返回值就是要遍历某一条边,但有返回值也要看如何处理返回值!
如果递归函数有返回值,如何区分要搜索一条边,还是搜索整个树呢?
搜索一条边的写法:
if (递归函数(root->left)) return ;if (递归函数(root->right)) return ;
搜索整个树写法:
left = 递归函数(root->left);right = 递归函数(root->right);left与right的逻辑处理;
看出区别了没?
在递归函数有返回值的情况下:如果要搜索一条边,递归函数返回值不为空的时候,立刻返回,如果搜索整个树,直接用一个变量left、right接住返回值,这个left、right后序还有逻辑处理的需要,也就是后序遍历中处理中间节点的逻辑(也是回溯)。
那么为什么要遍历整棵树呢?直观上来看,找到最近公共祖先,直接一路返回就可以了。
如图:
就像图中一样直接返回7,多美滋滋。
但事实上还要遍历根节点右子树(即使此时已经找到了目标节点了),也就是图中的节点4、15、20。
因为在如下代码的后序遍历中,如果想利用left和right做逻辑处理, 不能立刻返回,而是要等left与right逻辑处理完之后才能返回。
left = 递归函数(root->left);right = 递归函数(root->right);left与right的逻辑处理;
所以此时大家要知道我们要遍历整棵树。知道这一点,对本题就有一定深度的理解了。
那么先用left和right接住左子树和右子树的返回值,代码如下:
TreeNode* left = lowestCommonAncestor(root->left, p, q);TreeNode* right = lowestCommonAncestor(root->right, p, q);
如果left 和 right都不为空,说明此时root就是最近公共节点。这个比较好理解
如果left为空,right不为空,就返回right,说明目标节点是通过right返回的,反之依然。
这里有的同学就理解不了了,为什么left为空,right不为空,目标节点通过right返回呢?
如图:
图中节点10的左子树返回null,右子树返回目标值7,那么此时节点10的处理逻辑就是把右子树的返回值(最近公共祖先7)返回上去!
这里点也很重要,可能刷过这道题目的同学,都不清楚结果究竟是如何从底层一层一层传到头结点的。
那么如果left和right都为空,则返回left或者right都是可以的,也就是返回空。
代码如下:
if (left == NULL && right != NULL) return right;else if (left != NULL && right == NULL) return left;else { // (left == NULL && right == NULL)return NULL;}
那么寻找最小公共祖先,完整流程图如下:
从图中,大家可以看到,我们是如何回溯遍历整棵二叉树,将结果返回给头结点的!
整体代码如下:
class Solution {public:TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {if (root == q || root == p || root == NULL) return root;TreeNode* left = lowestCommonAncestor(root->left, p, q);TreeNode* right = lowestCommonAncestor(root->right, p, q);if (left != NULL && right != NULL) return root;if (left == NULL && right != NULL) return right;else if (left != NULL && right == NULL) return left;else { // (left == NULL && right == NULL)return NULL;}}};
稍加精简,代码如下:
class Solution {public:TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {if (root == q || root == p || root == NULL) return root;TreeNode* left = lowestCommonAncestor(root->left, p, q);TreeNode* right = lowestCommonAncestor(root->right, p, q);if (left != NULL && right != NULL) return root;if (left == NULL) return right;return left;}};
总结
这道题目刷过的同学未必真正了解这里面回溯的过程,以及结果是如何一层一层传上去的。
那么我给大家归纳如下三点:
- 求最小公共祖先,需要从底向上遍历,那么二叉树,只能通过后序遍历(即:回溯)实现从低向上的遍历方式。
- 在回溯的过程中,必然要遍历整棵二叉树,即使已经找到结果了,依然要把其他节点遍历完,因为要使用递归函数的返回值(也就是代码中的left和right)做逻辑判断。
- 要理解如果返回值left为空,right不为空为什么要返回right,为什么可以用返回right传给上一层结果。
可以说这里每一步,都是有难度的,都需要对二叉树,递归和回溯有一定的理解。
本题没有给出迭代法,因为迭代法不适合模拟回溯的过程。理解递归的解法就够了。
31、二叉树周末总结
32、二叉搜索树的最近公共祖先⭐
235. 二叉搜索树的最近公共祖先 - 力扣(LeetCode)

递归法
做过二叉树:公共祖先问题(opens new window)题目的同学应该知道,利用回溯从底向上搜索,遇到一个节点的左子树里有p,右子树里有q,那么当前节点就是最近公共祖先。
那么本题是二叉搜索树,二叉搜索树是有序的,那得好好利用一下这个特点。
在有序树里,如果判断一个节点的左子树里有p,右子树里有q呢?
其实只要从上到下遍历的时候,cur节点是数值在[p, q]区间中则说明该节点cur就是最近公共祖先了。
理解这一点,本题就很好解了。
和二叉树:公共祖先问题(opens new window)不同,普通二叉树求最近公共祖先需要使用回溯,从底向上来查找,二叉搜索树就不用了,因为搜索树有序(相当于自带方向),那么只要从上向下遍历就可以了。
那么我们可以采用前序遍历(其实这里没有中节点的处理逻辑,遍历顺序无所谓了)。
递归三部曲如下:
- 确定递归函数返回值以及参数
参数就是当前节点,以及两个结点 p、q。
返回值是要返回最近公共祖先,所以是TreeNode * 。
TreeNode* traversal(TreeNode* cur, TreeNode* p, TreeNode* q)
- 确定终止条件
其实都不需要这个终止条件,因为题目中说了p、q 为不同节点且均存在于给定的二叉搜索树中。也就是说一定会找到公共祖先的,所以并不存在遇到空的情况。
if (cur == NULL) return cur;
- 确定单层递归的逻辑
在遍历二叉搜索树的时候就是寻找区间[p->val, q->val](注意这里是左闭又闭)
那么如果 cur->val 大于 p->val,同时 cur->val 大于q->val,那么就应该向左遍历(说明目标区间在左子树上)。
需要注意的是此时不知道p和q谁大,所以两个都要判断
if (cur->val > p->val && cur->val > q->val) {TreeNode* left = traversal(cur->left, p, q);if (left != NULL) {return left;}}
细心的同学会发现,在这里调用递归函数的地方,把递归函数的返回值left,直接return。
在二叉树:公共祖先问题(opens new window)中,如果递归函数有返回值,如何区分要搜索一条边,还是搜索整个树。
搜索一条边的写法:
if (递归函数(root->left)) return ;if (递归函数(root->right)) return ;
搜索整个树写法:
left = 递归函数(root->left);right = 递归函数(root->right);left与right的逻辑处理;
本题就是标准的搜索一条边的写法,遇到递归函数的返回值,如果不为空,立刻返回。
如果 cur->val 小于 p->val,同时 cur->val 小于 q->val,那么就应该向右遍历(目标区间在右子树)。
if (cur->val < p->val && cur->val < q->val) {TreeNode* right = traversal(cur->right, p, q);if (right != NULL) {return right;}}
剩下的情况,就是cur节点在区间(p->val <= cur->val && cur->val <= q->val)或者 (q->val <= cur->val && cur->val <= p->val)中,那么cur就是最近公共祖先了,直接返回cur。
代码如下:
return cur;
那么整体递归代码如下:
/*** Definition for a binary tree node.* struct TreeNode {* int val;* TreeNode *left;* TreeNode *right;* TreeNode(int x) : val(x), left(NULL), right(NULL) {}* };*/class Solution {public:TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {if(root == NULL) return root;if(root->val > p->val && root->val > q->val){ // 左TreeNode* left = lowestCommonAncestor(root->left, p, q);if(left != NULL) return left;}if(root->val < p->val && root->val < q->val){ // 右TreeNode* right = lowestCommonAncestor(root->right, p, q);if(right != NULL) return right;}return root;}};
对代码进行简化
/*** Definition for a binary tree node.* struct TreeNode {* int val;* TreeNode *left;* TreeNode *right;* TreeNode(int x) : val(x), left(NULL), right(NULL) {}* };*/class Solution {public:TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {if(root == NULL) return root;if(root->val > p->val && root->val > q->val){if(left != NULL) return lowestCommonAncestor(root->left, p, q);}if(root->val < p->val && root->val < q->val){if(right != NULL) return lowestCommonAncestor(root->right, p, q);}return root;}};
迭代法
对于二叉搜索树的迭代法,大家应该在二叉树:二叉搜索树登场!(opens new window)就了解了。
利用其有序性,迭代的方式还是比较简单的,解题思路在递归中已经分析了。
迭代代码如下:
class Solution {public:TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {stack<TreeNode*> st;st.push(root);while(!st.empty()){TreeNode* cur = st.top(); // 中st.pop();if(cur->val > p->val && cur->val > q->val){ // 左st.push(cur->left);}else if(cur->val < p->val && cur->val < q->val){// 右st.push(cur->right);}else return cur;}return NULL;}};
这是我写的,果然还是功力不到位,其实可以利用辅助空间的。
看看卡哥写的
class Solution {public:TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {while(root){if(root->val > p->val && root->val > q->val){ // 左root = root->left;}else if(root->val < p->val && root->val < q->val){// 右root = root->right;}else return root;}return NULL;}};
什么他妈的叫极简!
总结
对于二叉搜索树的最近祖先问题,其实要比普通二叉树公共祖先问题(opens new window)简单的多。
不用使用回溯,二叉搜索树自带方向性,可以方便的从上向下查找目标区间,遇到目标区间内的节点,直接返回。
最后给出了对应的迭代法,二叉搜索树的迭代法甚至比递归更容易理解,也是因为其有序性(自带方向性),按照目标区间找就行了。
33、二叉搜索树中的插入操作
701. 二叉搜索树中的插入操作 - 力扣(LeetCode)


递归法
其实这道题目其实是一道简单题目,但是题目中的提示:有多种有效的插入方式,还可以重构二叉搜索树,一下子吓退了不少人,瞬间感觉题目复杂了很多。
其实可以不考虑题目中提示所说的改变树的结构的插入方式。
如下演示视频中可以看出:只要按照二叉搜索树的规则去遍历,遇到空节点就插入节点就可以了。
例如插入元素10 ,需要找到末尾节点插入便可,一样的道理来插入元素15,插入元素0,插入元素6,需要调整二叉树的结构么? 并不需要。。
只要遍历二叉搜索树,找到空节点 插入元素就可以了,那么这道题其实就简单了。
接下来就是遍历二叉搜索树的过程了。
- 确定递归函数的参数以及返回值
参数就是根节点指针,以及要插入元素,这里递归函数要不要有返回值呢?
可以有,也可以没有,但递归函数如果没有返回值的话,实现是比较麻烦的,下面也会给出其具体实现代码。
有返回值的话,可以利用返回值完成新加入的节点与其父节点的赋值操作。(下面会进一步解释)
- 确定终止条件
终止条件就是找到遍历的节点为null的时候,就是要插入节点的位置了,并把插入的节点返回。
- 确定单层递归的逻辑
此时要明确,需要遍历整棵树么?
别忘了这是搜索树,遍历整棵搜索树简直是对搜索树的侮辱,哈哈。
搜索树是有方向了,可以根据插入元素的数值,决定递归方向。
完整C++代码如下:
/*** 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:TreeNode* insertIntoBST(TreeNode* root, int val) {if(root == nullptr){ // 遍历的节点为null的时候,就是要插入节点的位置TreeNode* node = new TreeNode(val);return node;};if(root->val > val) root->left = insertIntoBST(root->left, val);if(root->val < val) root->right = insertIntoBST(root->right, val);return root;}};
到这里,大家应该能感受到,如何通过递归函数返回值完成了新加入节点的父子关系赋值操作了,下一层将加入节点返回,本层用root->left或者root->right将其接住。
刚刚说了递归函数不用返回值也可以,找到插入的节点位置,直接让其父节点指向插入节点,结束递归,也是可以的。
那么递归函数定义如下:
TreeNode* parent; // 记录遍历节点的父节点void traversal(TreeNode* cur, int val)
没有返回值,需要记录上一个节点(parent),遇到空节点了,就让parent左孩子或者右孩子指向新插入的节点。然后结束递归。
/*** 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:TreeNode* parent; // 要插入节点的位置的父节点void traversal(TreeNode* root, int val){if(root == nullptr){TreeNode* node = new TreeNode(val);if(parent && val > parent->val) parent->right = node;else if(parent && val < parent->val) parent->left = node;return;}parent = root;if(root->val > val) traversal(root->left, val);if(root->val < val) traversal(root->right, val);return;}TreeNode* insertIntoBST(TreeNode* root, int val) {if(root == nullptr){root = new TreeNode(val);}traversal(root, val);return root;}};
可以看出还是麻烦一些的。
我之所以举这个例子,是想说明通过递归函数的返回值完成父子节点的赋值是可以带来便利的。
网上千变一律的代码,可能会误导大家认为通过递归函数返回节点 这样的写法是天经地义,其实这里是有优化的!
迭代法
再来看看迭代法,对二叉搜索树迭代写法不熟悉,可以看这篇:二叉树:二叉搜索树登场!(opens new window)
在迭代法遍历的过程中,需要记录一下当前遍历的节点的父节点,这样才能做插入节点的操作。
在二叉树:搜索树的最小绝对差(opens new window)和二叉树:我的众数是多少?(opens new window)中,都是用了记录pre和cur两个指针的技巧,本题也是一样的。
代码如下:
/*** 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:TreeNode* insertIntoBST(TreeNode* root, int val) {if(root == nullptr) return new TreeNode(val);TreeNode* temp = root;TreeNode* parent; // 记录前一个节点while(temp){parent = temp;if(temp->val > val){temp = temp->left;if(temp == nullptr) parent->left = new TreeNode(val);}else if(temp->val < val){temp = temp->right;if(temp == nullptr) parent->right = new TreeNode(val);}}return root;}};
总结
首先在二叉搜索树中的插入操作,大家不用恐惧其重构搜索树,其实根本不用重构。
然后在递归中,我们重点讲了如果通过递归函数的返回值完成新加入节点和其父节点的赋值操作,并强调了搜索树的有序性。
最后依然给出了迭代的方法,迭代的方法就需要记录当前遍历节点的父节点了,这个和没有返回值的递归函数实现的代码逻辑是一样的。
34、删除二叉搜索树中的节点
450. 删除二叉搜索树中的节点 - 力扣(LeetCode)


递归法
递归三部曲:
- 确定递归函数参数以及返回值
说道递归函数的返回值,在二叉树:搜索树中的插入操作(opens new window)中通过递归返回值来加入新节点, 这里也可以通过递归返回值删除节点。
- 确定终止条件
遇到空返回,其实这也说明没找到删除的节点,遍历到空节点直接返回了
- 确定单层递归的逻辑
这里就把二叉搜索树中删除节点遇到的情况都搞清楚。
有以下五种情况:
- 第一种情况:没找到删除的节点,遍历到空节点直接返回了
- 找到删除的节点
- 第二种情况:左右孩子都为空(叶子节点),直接删除节点, 返回NULL为根节点
- 第三种情况:删除节点的左孩子为空,右孩子不为空,删除节点,右孩子补位,返回右孩子为根节点
- 第四种情况:删除节点的右孩子为空,左孩子不为空,删除节点,左孩子补位,返回左孩子为根节点
- 第五种情况:左右孩子节点都不为空,则将删除节点的左子树头结点(左孩子)放到删除节点的右子树的最左面节点的左孩子上,返回删除节点右孩子为新的根节点。
第五种情况有点难以理解,看下面动画:
class Solution {public:TreeNode* deleteNode(TreeNode* root, int key) {if (root == nullptr) return root; // 第一种情况:没找到删除的节点,遍历到空节点直接返回了if (root->val == key) {// 第二种情况:左右孩子都为空(叶子节点),直接删除节点, 返回NULL为根节点if (root->left == nullptr && root->right == nullptr) {///! 内存释放delete root;return nullptr;}// 第三种情况:其左孩子为空,右孩子不为空,删除节点,右孩子补位 ,返回右孩子为根节点else if (root->left == nullptr) {auto retNode = root->right;///! 内存释放delete root;return retNode;}// 第四种情况:其右孩子为空,左孩子不为空,删除节点,左孩子补位,返回左孩子为根节点else if (root->right == nullptr) {auto retNode = root->left;///! 内存释放delete root;return retNode;}// 第五种情况:左右孩子节点都不为空,则将删除节点的左子树放到删除节点的右子树的最左面节点的左孩子的位置// 并返回删除节点右孩子为新的根节点。else {TreeNode* cur = root->right; // 找右子树最左面的节点while(cur->left != nullptr) {cur = cur->left;}cur->left = root->left; // 把要删除的节点(root)左子树放在cur的左孩子的位置TreeNode* tmp = root; // 把root节点保存一下,下面来删除root = root->right; // 返回旧root的右孩子作为新rootdelete tmp; // 释放节点内存(这里不写也可以,但C++最好手动释放一下吧)return root;}}if (root->val > key) root->left = deleteNode(root->left, key);if (root->val < key) root->right = deleteNode(root->right, key);return root;}};
普通二叉树的删除方式
这里我在介绍一种通用的删除,普通二叉树的删除方式(没有使用搜索树的特性,遍历整棵树),用交换值的操作来删除目标节点。
代码中目标节点(要删除的节点)被操作了两次:
- 第一次是和目标节点的右子树最左面节点交换。
- 第二次直接被NULL覆盖了。
思路有点绕,感兴趣的同学可以画图自己理解一下。
代码如下:(关键部分已经注释)
class Solution {public:TreeNode* deleteNode(TreeNode* root, int key) {if (root == nullptr) return root;if (root->val == key) {if (root->right == nullptr) { // 这里第二次操作目标值:最终删除的作用return root->left;}TreeNode *cur = root->right;while (cur->left) {cur = cur->left;}swap(root->val, cur->val); // 这里第一次操作目标值:交换目标值其右子树最左面节点。}root->left = deleteNode(root->left, key);root->right = deleteNode(root->right, key);return root;}};
有点难,还是搞不明白!
迭代法
删除节点的迭代法还是复杂一些的,但其本质我在递归法里都介绍了,最关键就是删除节点的操作(动画模拟的过程)
代码如下:
/*** 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:// 将目标节点(删除节点)的左子树放到 目标节点的右子树的最左面节点的左孩子位置上// 并返回目标节点右孩子为新的根节点// 是动画里模拟的过程TreeNode* deleteOneNode(TreeNode* target){if(target == nullptr) return target;if(target->right == nullptr) return target->left;if(target->left == nullptr) return target->right; // 其实后序的逻辑已经包含了这里的逻辑判断了,这里省略也是可以的TreeNode* cur = target->right;while(cur->left){cur = cur->left;}cur->left = target->left;return target->right;}TreeNode* deleteNode(TreeNode* root, int key) {if(root == nullptr) return root;TreeNode* cur = root;TreeNode* pre = nullptr; // 记录cur的父节点,用来删除curwhile(cur){if(cur->val == key) break;pre = cur;if(cur->val > key) cur = cur->left;else cur = cur->right;}if(pre == nullptr){ // 如果搜索树只有头结点return deleteOneNode(cur);}// pre 要知道是删左孩子还是右孩子if(pre->left && pre->left->val == key){pre->left = deleteOneNode(cur);}if(pre->right && pre->right->val == key){pre->right = deleteOneNode(cur);}return root;}};
35、修剪二叉搜索树
递归法
直接想法就是:递归处理,然后遇到 root->val < low || root->val > high 的时候直接return NULL,一波修改,赶紧利落。
不难写出如下代码:
class Solution {public:TreeNode* trimBST(TreeNode* root, int low, int high) {if (root == nullptr || root->val < low || root->val > high) return nullptr;root->left = trimBST(root->left, low, high);root->right = trimBST(root->right, low, high);return root;}};
然而[1, 3]区间在二叉搜索树的中可不是单纯的节点3和左孩子节点0就决定的,还要考虑节点0的右子树。
从图中可以看出需要重构二叉树,想想是不是本题就有点复杂了。
其实不用重构那么复杂。
在上图中我们发现节点0并不符合区间要求,那么将节点0的右孩子 节点2 直接赋给 节点3的左孩子就可以了(就是把节点0从二叉树中移除),如图:
这道题可以有返回值,也可以没有返回值,但是有返回值处理起来会比较方便。
完整C++代码如下:
/*** 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:TreeNode* trimBST(TreeNode* root, int low, int high) {if(root == nullptr) return root;if(root->val < low){ // 当前root->val < low, 应该递归右子树TreeNode* right = trimBST(root->right, low, high);return right;}if(root->val > high){ // 当前root->val > high, 应该递归左子树TreeNode* left = trimBST(root->left, low, high);return left;}root->left = trimBST(root->left, low, high);root->right = trimBST(root->right, low, high);return root;}};
迭代法
因为二叉搜索树的有序性,不需要使用栈模拟递归的过程。(以为做题的经验,需要改变树的结构的,一般都不用使用栈来模拟递归)
在剪枝的时候,可以分为三步:
- 将root移动到[L, R] 范围内,注意是左闭右闭区间
- 剪枝左子树
- 剪枝右子树
代码如下:
/*** 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:TreeNode* trimBST(TreeNode* root, int low, int high) {if(root == nullptr) return root;// 处理头节点,让root移动到[low, high]范围内while(root != nullptr && (root->val < low || root->val > high)){if(root->val < low) root = root->right; // 小于往右走else root = root->left; // 大于往左走}TreeNode* cur = root;// 此时root已经在[low, high] 范围内,处理左孩子元素小于low的情况while(cur != nullptr){while(cur->left && cur->left->val < low){cur->left = cur->left->right;}cur = cur->left; // 遍历左节点}cur = root;// 此时root已经在[low, high] 范围内,处理右孩子元素大于high的情况while(cur != nullptr){while(cur->right && cur->right->val > high){cur->right = cur->right->left;}cur = cur->right; // 遍历右节点}return root;}};
总结
修剪二叉搜索树其实并不难,但在递归法中大家可看出我费了很大的功夫来讲解如何删除节点的,这个思路其实是比较绕的。
最终的代码倒是很简洁。
如果不对递归有深刻的理解,这道题目还是有难度的!
本题我依然给出递归法和迭代法,初学者掌握递归就可以了,如果想进一步学习,就把迭代法也写一写。
36、将有序数组转换为二叉搜索树
108. 将有序数组转换为二叉搜索树 - 力扣(LeetCode)


题目中说要转换为一棵高度平衡二叉搜索树。这和转换为一棵普通二叉搜索树有什么差别呢?
其实这里不用强调平衡二叉搜索树,数组构造二叉树,构成平衡树是自然而然的事情,因为大家默认都是从数组中间位置取值作为节点元素,一般不会随机取,所以想构成不平衡的二叉树是自找麻烦。
在二叉树:构造二叉树登场!(opens new window)和二叉树:构造一棵最大的二叉树(opens new window)中其实已经讲过了,如果根据数组构造一棵二叉树。
本质就是寻找分割点,分割点作为当前节点,然后递归左区间和右区间。
本题其实要比二叉树:构造二叉树登场!(opens new window)和 二叉树:构造一棵最大的二叉树(opens new window)简单一些,因为有序数组构造二叉搜索树,寻找分割点就比较容易了。
分割点就是数组中间位置的节点。
那么为问题来了,如果数组长度为偶数,中间节点有两个,取哪一个?
取哪一个都可以,只不过构成了不同的平衡二叉搜索树。
递归法
利用了容器,开销较大
class Solution {public:TreeNode* sortedArrayToBST(vector<int>& nums) {// 如果数组为空,返回nullptrif(nums.size() == 0) return nullptr;int delimiterindex = nums.size() / 2; // 数组最中间的元素,就是要分割的节点TreeNode* root = new TreeNode(nums[delimiterindex]);// 左闭右开vector<int> leftNums(nums.begin(), nums.begin() + delimiterindex);vector<int> rightNums(nums.begin() + delimiterindex + 1, nums.end());root->left = sortedArrayToBST(leftNums);root->right = sortedArrayToBST(rightNums);return root;}};
可以对代码进行优化
class Solution {public:TreeNode* traversal(vector<int>& nums, int begin, int end){// 如果数组为空,返回nullptrif(begin > end) return nullptr;int mid = begin + (end - begin) / 2; // 数组最中间的元素,就是要分割的节点TreeNode* root = new TreeNode(nums[mid]);// 左闭右开int leftBegin = begin;int leftEnd = mid - 1;int rightBegin = mid + 1;int rightEnd = end;// 递归处理root->left = traversal(nums, leftBegin, leftEnd);root->right = traversal(nums, rightBegin, rightEnd);return root;}TreeNode* sortedArrayToBST(vector<int>& nums) {return traversal(nums, 0, nums.size() - 1);}};
代码可以进一步简化,直接这样写:
// 递归处理root->left = traversal(nums, begin, mid - 1);root->right = traversal(nums, mid + 1, end);
但是我觉得没必要,上面那样子才能充分体现出解题的逻辑。
迭代法
迭代法可以通过三个队列来模拟,一个队列放遍历的节点,一个队列放左区间下标,一个队列放右区间下标。
模拟的就是不断分割的过程,C++代码如下:(我已经详细注释)
/*** 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:TreeNode* sortedArrayToBST(vector<int>& nums) {if(nums.size() == 0) return nullptr;TreeNode* root = new TreeNode(0); // 初始化根节点queue<TreeNode*> nodeQue; // 放遍历的节点queue<int> leftQue; // 保存左区间下标queue<int> rightQue; // 保存右区间下标nodeQue.push(root); // 根节点入队列leftQue.push(0); // 0为左区间下标初始化位置rightQue.push(nums.size() - 1); // nums.size() - 1为右区间下标初始位置while(!nodeQue.empty()){TreeNode* cur = nodeQue.front();nodeQue.pop();int left = leftQue.front(); leftQue.pop();int right = rightQue.front(); rightQue.pop();int mid = left + ((right - left) >> 1);cur->val = nums[mid]; // 将mid对应的元素给中间节点if(left <= mid - 1){ // 处理左节点cur->left = new TreeNode(0);nodeQue.push(cur->left);leftQue.push(left);rightQue.push(mid - 1);}if(right >= mid + 1){ // 处理右节点cur->right = new TreeNode(0);nodeQue.push(cur->right);leftQue.push(mid + 1);rightQue.push(right);}}return root;}};
总结
在二叉树:构造二叉树登场!(opens new window)和 二叉树:构造一棵最大的二叉树(opens new window)之后,我们顺理成章的应该构造一下二叉搜索树了,一不小心还是一棵平衡二叉搜索树。
其实思路也是一样的,不断中间分割,然后递归处理左区间,右区间,也可以说是分治。
此时相信大家应该对通过递归函数的返回值来增删二叉树很熟悉了,这也是常规操作。
在定义区间的过程中我们又一次强调了循环不变量的重要性。
最后依然给出迭代的方法,其实就是模拟取中间元素,然后不断分割去构造二叉树的过程。
37、把二叉树转换为累加树
累加树的要求:当前节点的值 = 原来的值 + 二叉搜索树中所有节点值大于它的值
所以,很明显了,遍历的顺序应该是:右—>中—>左,而中序遍历的顺序是 左中右,因此把这种解法叫做反中序遍历法
递归法
/*** 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:TreeNode* pre = nullptr; // 记录最近访问过的节点void traversal(TreeNode* root){if(root == nullptr) return;traversal(root->right); // 右if(pre) root->val += pre->val; // 中pre = root; // 更新pretraversal(root->left); // 左}TreeNode* convertBST(TreeNode* root) {traversal(root);return root;}};
迭代法
/*** 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:TreeNode* convertBST(TreeNode* root) {if(root == nullptr) return root;stack<TreeNode*> st;st.push(root);TreeNode* pre = nullptr; // 记录最近一次访问过的节点while(!st.empty()){TreeNode* cur = st.top();if(cur){st.pop();if(cur->left) st.push(cur->left); // 左st.push(cur); // 中st.push(nullptr); // 插入标记if(cur->right) st.push(cur->right); // 右}else{st.pop();cur = st.top();if(pre != nullptr) cur->val += pre->val;pre = cur;st.pop();}}return root;}};














































