系列目录
88.合并两个有序数组
52.螺旋数组
567.字符串的排列
643.子数组最大平均数
150.逆波兰表达式
61.旋转链表
160.相交链表
83.删除排序链表中的重复元素
389.找不同
1491.去掉最低工资和最高工资后的工资平均值
896.单调序列
206.反转链表
92.反转链表II
141.环形链表
142.环型链表
21.合并两个有序列表
24.两辆交换链表中的节点
876.链表的中间节点
143. 重排链表
2.两数相加
445.两数相加II
前言
方法都是类似的~
递归是隐式调用栈(递归栈,无需手动实现),迭代是显式调用栈
Morris遍历
简介
Morris遍历是一种高效的二叉树遍历算法,其显著特点是能在不使用额外的栈或队列,而是利用树大量的空闲指针完成遍历,空间复杂度为O(1)
1. 遍历思路
1、开始时 cur = root,cur为空时过程停止
2、若cur无左树,cur向右移动
3、若cur有左树,找到左树最右节点mostRight
a.若mostRight的右指针指向空,让其指向cur,cur向左移动
b.若mostRight的右指针指向cur,让其指向空,cur向右移动
举例:
/** a
* / \
* b e
* / \ / \
* c d f g
*
* Morris = [a, b, c, b, d, a, e, f, e, g]
/
2. 类型转换
Morris遍历可以实现前序、中序和后序遍历
- 前序遍历:在第一次访问节点时,输出节点的值,然后进行左子树的遍历
- 中序遍历:在第二次访问节点时(即mostRight的右指针指向当前节点时),输出节点的值,然后进行右子树的遍历
- 后序遍历:后序遍历的实现相对复杂,需要在遍历过程中记录左子树最右分支的路径,并在回溯时逆序输出
3. 优缺点
- 优点:Morris遍历的空间复杂度为O(1),比使用栈或递归的遍历算法更高效
此外,Morris遍历不会占用额外的内存空间,对于内存受限的环境非常友好 - 缺点:Morris遍历会修改原始树的结构,需要在遍历完成后还原 此外,Morris遍历的实现相对复杂,需要理解其背后的思想和原理
代码实现
// Morris遍历
public:
vector<int> void morris(TreeNode *root) {
TreeNode *cur = root;
TreeNode *mostRight = nullptr;
while (cur != cullptr) {
mostRight = cur->left;
// cur 有左树
if (mostRight != nullptr) {
// 最右节点不为空 且 不为cur
while (mostRight->right != nullptr && mostRight->right != cur){
most = most->right;
}
// 第一次到达
// cur左子树的 最右节点mostRight的右指针指向空
// cur 向右移动
if (mostRight == nullptr) {
mostRight->right = cur;
cur = cur->left;
continue;
}
// 第二次到达
// 此时cur左子树的 最右节点mostRight的右指针指向cur
// cur 向左移动
else {
// 恢复
mostRight->right = nullptr;
// cur = cur->right;
}
}
// cur 无子树,向右移动
cur = cur->right;
}
};
Morris遍历模板
public:
vector<int> marris(TreeNode *root) {
TreeNode *cur = root;
TreeNode *mostRight = nullptr;
while (cur != nullptr) {
mostRight = cur->left;
if (mostRight != nullptr) {
while (mostRight->right != nullptr && mostRight->right != cur) {
mostRight = mostRight->right;
}
if (mostRight == nullptr) {
mostRight->right = cur;
cur = cur->left;
continue;
} else {
mostRight->right = nullptr;
// cur = cur->right
}
}
cur = cur->right;
}
};
Morris遍历加工出前序
/**
* 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) {
// 修改I
vector<int> res;
if (root == nullptr)
return res;
TreeNode *cur = root;
TreeNode *mostRight = nullptr;
while (cur != nullptr) {
mostRight = cur->left;
// cur 有左树
if (mostRight != nullptr) {
// 最右节点不为空 且 不为cur
while (mostRight->right != nullptr && mostRight->right != cur){
mostRight = mostRight->right;
}
// 第一次到达
// cur左子树的 最右节点mostRight的右指针指向空
// cur 向右移动
if (mostRight->right == nullptr) {
// 对于有左子树的节点——第一次收集
// 修改II
// emplace_back()也可
res.push_back(cur->val);
mostRight->right = cur;
cur = cur->left;
// 别忘了 continue
continue;
}
// 第二次到达
// cur左子树的 最右节点mostRight的右指针指向cur
// cur 向左移动
else {
// 恢复
mostRight->right = nullptr;
// cur = cur->right;
}
} else {
// 对于没有左子树的节点——直接收集
// 修改III
res.emplace_back(cur->val);
}
// cur 无子树,向右移动
cur = cur->right;
}
// 修改IV
// 别忘了 修改时在最后返回res
return res;
}
};
Morris遍历加工出中序
写法一 一般写法
/**
* 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) {
// 修改I
vector<int> ans;
if (root == nullptr)
return ans;
TreeNode *cur = root, *mostRight = nullptr;
while (cur != nullptr) {
mostRight = cur->left;
if (mostRight != nullptr) {
while (mostRight->right != nullptr && mostRight->right != cur)
mostRight = mostRight->right;
if (mostRight->right == nullptr) {
mostRight->right = cur;
cur = cur->left;
continue;
} else {
// 修改II
ans.push_back(cur->val);
mostRight->right = nullptr;
}
} else {
// 修改III
ans.push_back(cur->val);
}
cur = cur->right;
}
// 修改IV
return ans;
}
};
写法二 简略写法
/**
* 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) {
// 修改I
vector<int> ans;
if (root == nullptr)
return ans;
TreeNode *cur = root, *mostRight = nullptr;
while (cur != nullptr) {
mostRight = cur->left;
if (mostRight != nullptr) {
while (mostRight->right != nullptr && mostRight->right != cur)
mostRight = mostRight->right;
if (mostRight->right == nullptr) {
mostRight->right = cur;
cur = cur->left;
continue;
} else {
mostRight->right = nullptr;
}
}
// 修改II
// 无左树——只被遍历1次;有左树——第2次遍历 都会经过这里
ans.push_back(cur->val);
cur = cur->right;
}
// 修改III
return ans;
}
};
144.二叉树的前序遍历
🌟递归
C++
若未特殊标明,以下题解均写用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:
// 注意引用符
void preorder(TreeNode* root, vector<int> &res) {
if (root == nullptr)
// preorder 函数被定义为返回 void 类型——不返回任何值
// return; 仅仅是结束函数的执行,并不返回任何值
return;
// 存入向量的末尾
res.push_back(root->val);
// 前序——根左右
preorder(root->left, res);
preorder(root->right, res);
}
vector<int> preorderTraversal(TreeNode* root) {
// 定义一个 矢量/向量——更准确地说是一个动态数组
vector<int> res;
preorder(root, res);
return res;
}
};
方法二 Morris遍历
/**
* 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) {
// 修改I
vector<int> res;
if (root == nullptr)
return res;
TreeNode *cur = root;
TreeNode *mostRight = nullptr;
while (cur != nullptr) {
mostRight = cur->left;
// cur 有左树
if (mostRight != nullptr) {
// 最右节点不为空 且 不为cur
while (mostRight->right != nullptr && mostRight->right != cur){
mostRight = mostRight->right;
}
// 第一次到达
// cur左子树的 最右节点mostRight的右指针指向空
// cur 向右移动
if (mostRight->right == nullptr) {
// 对于有左子树的节点——第一次收集
// 修改II
// emplace_back()也可
res.push_back(cur->val);
mostRight->right = cur;
cur = cur->left;
// 别忘了 continue
continue;
}
// 第二次到达
// cur左子树的 最右节点mostRight的右指针指向cur
// cur 向左移动
else {
// 恢复
mostRight->right = nullptr;
// cur = cur->right;
}
} else {
// 对于没有左子树的节点——直接收集
// 修改III
res.emplace_back(cur->val);
}
// cur 无子树,向右移动
cur = cur->right;
}
// 修改IV
// 别忘了 修改时在最后返回res
return res;
}
};
94.二叉树的中序遍历
🌟递归+迭代
C++
若未特殊标明,以下题解均写用C++
方法一 递归 (DFS )
/**
* Definition for a binary tree node.
* struct TreeNode {
* int val;
* TreeNode *left;
* TreeNode *right;
* TreeNode() : val(0), left(nullptr), right(nullptr) {}
* TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
* TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
* };
*/
// 二叉树的中序遍历
class Solution {
public:
void inorder(TreeNode* root, vector<int>& res) {
//可以另写成if (!root)
if (!root)
return;
// 中序——左根右——先插入最左边的左孩子
inorder(root->left, res);
res.push_back(root->val);
inorder(root->right, res);
}
vector<int> inorderTraversal(TreeNode* root) {
vector<int> res;
inorder(root, res);
return res;
}
};
方法二 Morris遍历
/**
* 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) {
// 修改I
vector<int> ans;
if (root == nullptr)
return ans;
TreeNode *cur = root, *mostRight = nullptr;
while (cur != nullptr) {
mostRight = cur->left;
if (mostRight != nullptr) {
while (mostRight->right != nullptr && mostRight->right != cur)
mostRight = mostRight->right;
if (mostRight->right == nullptr) {
mostRight->right = cur;
cur = cur->left;
continue;
} else {
// 修改II
ans.push_back(cur->val);
mostRight->right = nullptr;
}
} else {
// 修改III
ans.push_back(cur->val);
}
cur = cur->right;
}
// 修改IV
return ans;
}
};
145.二叉树的后序遍历
🌟递归+迭代
C++
若未特殊标明,以下题解均写用C++
方法一 递归 (DFS )
/**
* Definition for a binary tree node.
* struct TreeNode {
* int val;
* TreeNode *left;
* TreeNode *right;
* TreeNode() : val(0), left(nullptr), right(nullptr) {}
* TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
* TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
* };
*/
// 二叉树的后序遍历
class Solution {
public:
void postorder(TreeNode *root, vector<int> &res) {
if (!root)
return;
// 后序——左右根
postorder(root->left, res);
postorder(root->right, res);
// 左右都不存在才先插入根节点的值
res.push_back(root->val);
}
vector<int> postorderTraversal(TreeNode* root) {
vector<int> res;
postorder(root, res);
return res;
}
};
方法二 迭代
/**
* 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> res;
// 空树
if (!root)
return res;
// 定义一个 树节点栈
stack<TreeNode* > stk;
TreeNode *prev = nullptr;
// 树没遍历完 或 栈非空
while (root != nullptr || !stk.empty()) {
// 树没遍历完
while (root != nullptr) {
// 遍历所有左节点 入栈
// 将这个树节点 入栈
stk.emplace(root);
root = root->left;
}
// 若此时 root 为空
// 更新 root
root = stk.top();
// 栈顶元素用完 弹出栈
stk.pop();
// 无右子节点 或 这个右子节点被遍历过
if (root->right == nullptr || root->right == prev) {
res.emplace_back(root->val);
// 标记这个节点
prev = root;
root = nullptr;
}
// 有右子节点
else {
stk.emplace(root);
root = root->right;
}
}
return res;
}
};
方法三 Morris遍历
class Solution {
public:
void addPath(vector<int> &vec, TreeNode *node) {
int count = 0;
while (node != nullptr) {
++count;
vec.emplace_back(node->val);
node = node->right;
}
reverse(vec.end() - count, vec.end());
}
vector<int> postorderTraversal(TreeNode *root) {
vector<int> res;
if (root == nullptr) {
return res;
}
TreeNode *p1 = root, *p2 = nullptr;
while (p1 != nullptr) {
p2 = p1->left;
if (p2 != nullptr) {
while (p2->right != nullptr && p2->right != p1) {
p2 = p2->right;
}
if (p2->right == nullptr) {
p2->right = p1;
p1 = p1->left;
continue;
} else {
p2->right = nullptr;
addPath(res, p1->left);
}
}
p1 = p1->right;
}
addPath(res, root);
return res;
}
};
102.二叉树的层序遍历
🌟BFS+队列
C++
若未特殊标明,以下题解均写用C++
方法一 BFS
/**
* 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>> ret;
if (!root) {
return ret;
}
queue <TreeNode*> q;
q.push(root);
while (!q.empty()) {
int currentLevelSize = q.size();
ret.push_back(vector <int> ());
for (int i = 1; i <= currentLevelSize; ++i) {
auto node = q.front(); q.pop();
ret.back().push_back(node->val);
if (node->left) q.push(node->left);
if (node->right) q.push(node->right);
}
}
return ret;
}
};
方法二 队列
/**
* 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) {}
* };
*/
// #include <queue>
// #include <vector>
class Solution {
public:
vector<vector<int>> levelOrder(TreeNode* root) {
vector<vector<int>> ans;
if (root == nullptr )
return ans;
queue<TreeNode*> cur;
cur.push(root);
while (!cur.empty()) {
// 当前层的节点数
int size = cur.size();
// 存储当前层的节点值
vector<int> vals;
for (int i = 0; i < size; ++i) {
TreeNode* node = cur.front();
cur.pop();
vals.push_back(node->val);
if (node->left)
cur.push(node->left);
if (node->right)
cur.push(node->right);
}
ans.push_back(vals); // 将当前层的节点值添加到答案中
}
return ans;
}
};