Last updated on 9 months ago
给你二叉搜索树的根节点 root
,该树中的 恰好 两个节点的值被错误地交换。请在不改变其结构的情况下,恢复这棵树 。
思路: 二叉搜索树 中序排序是升序的,中序遍历时 记录前一个节点, 比较当前节点和上个节点,正常的树是当前节点要大于上一节点
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
| class Solution { public: TreeNode *t = nullptr; TreeNode *a = nullptr; TreeNode *b = nullptr; void backTree(TreeNode *root) { if (root->left != nullptr) backTree(root->left); if (t != nullptr && t->val > root->val) { if (a == nullptr) { a = t; } b = root; } t = root; if (root->right != nullptr) { backTree(root->right); } } void recoverTree(TreeNode *root) { backTree(root); swap(a->val,b->val); } };
|
2 .树的前中后序 遍历
递归排序: 相对于递归比较简单,总体公式就是
- 确定递归的参数,返回值
- 递归终止的条件
- 递归的逻辑 (前中还是先后)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
|
void traversal(TreeNode* cur, vector<int>& vec) { if (cur == NULL) return; traversal(cur->left, vec); vec.push_back(cur->val); traversal(cur->right, vec); }
void traversal(TreeNode* cur, vector<int>& vec) { if (cur == NULL) return; traversal(cur->left, vec); traversal(cur->right, vec); vec.push_back(cur->val); }
|
递归可以的,用迭代一般都可以都可以:
使用一个栈来辅助
- 前序遍历: 遍历第一个数进栈,然后出栈,出栈前,先看下他有无左右子树,有的话先进右节点,后进左节点,为什么呢?因为前序是先忘左边遍历的后面才往右边遍历
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| class Solution { public: vector<int> preorderTraversal(TreeNode* root) { stack<TreeNode*> st; vector<int> result; if (root == NULL) return result; st.push(root); while (!st.empty()) { TreeNode* node = st.top(); st.pop(); result.push_back(node->val); if (node->right) st.push(node->right); if (node->left) st.push(node->left); } return result; } };
|
- 中序遍历: 对于中序,有点特殊,遍历的结果是左右都没有节点的时候才是我们想要的节点,思路还是一样先一直往左走,走一个压栈一个,遇到没有左子节点的时候,看下有无右节点,没有则出栈(左中右),看其右边有无子节点
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
| class Solution { public: vector<int> inorderTraversal(TreeNode* root) { vector<int> result; stack<TreeNode*> st; TreeNode* cur = root; while (cur != NULL || !st.empty()) { if (cur != NULL) { st.push(cur); cur = cur->left; } else { cur = st.top(); st.pop(); result.push_back(cur->val); cur = cur->right; } } return result; } };
|
后序遍历: 前序遍历时是 中左右 ,当然我们也可以中右左(改变遍历顺序),将这个结果倒序排一遍就是后序遍历了…
给你二叉树的根节点 root
,返回其节点值的 层序遍历 。 (即逐层地,从左到右访问所有节点)。
![image-20220321120854055](http://imt.rui.vin/202203211218012.png)
思路: 会这题可以解决很多树的问题!
- 利用队列的性质,第一次进3,出队列,然后3的左右子树依次进队,此时 队列里有9,10,让后重复出一个节点,然后该节点的左右节点分别进入,同理,10 节点也是一样
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
| class Solution { public: vector<vector<int>> levelOrder(TreeNode* root) { queue<TreeNode *> que; if(root != nullptr) que.push(root); vector<vector<int>> res; vector<int> num; while (!que.empty()) { int size = que.size(); for (int i = 0; i < size; i++) { TreeNode* node = que.front(); que.pop(); num.push_back(node->val); if (node->left) que.push(node->left); if (node->right) que.push(node->right); } res.push_back(num); num.clear(); } return res; } };
|
- 层次遍历可以解决 二叉树的层数也是可以解决,加一个变量记录进入for的次数,而且还可以记录出最小的高度,即节点的左右子节点都为空的时候
给定两个整数数组 preorder 和 inorder ,其中 preorder 是二叉树的先序遍历, inorder 是同一棵树的中序遍历,请构造二叉树并返回其根节点。
已知前序和中序 推后序,很基础的题,但是用代码 实现就有点困难了,应该是用递归,不知怎么递归,以及边界的 处理
前序的 每个数都将中序分为两部分,通过中序左右两部分个数 又可以将前序分为两部分
·
通过前序的值 可以找到中序的下标索引 然后 在开始划分,左子树 递归 和 右子树 递归
为了方便通过前序值马上找到中序的下标,可以先用 map来索引
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
| class Solution { public: map<int,int> nodemap; TreeNode* drawback(vector<int>& preorder, vector<int>& inorder,int Pleft,int Pright,int Ileft,int Iright){ if(Pleft > Pright) return nullptr; int preroot = preorder[Pleft]; int inroot = nodemap[preroot]; TreeNode *root = new TreeNode(preroot);
int size_left_node = inroot - Ileft; root->left = drawback(preorder,inorder,Pleft+1,Pleft+size_left_node,Ileft,inroot-1); root->right = drawback(preorder,inorder,Pleft+size_left_node+1,Pright,inroot+1,Iright); return root; } TreeNode* buildTree(vector<int>& preorder, vector<int>& inorder) { if(!preorder.size() || !inorder.size()){ return nullptr; } for(int i = 0; i < inorder.size();i++ ){ nodemap[inorder[i]] = i; }
return drawback(preorder,inorder,0,inorder.size()-1,0,inorder.size()-1); } };
|
路径 被定义为一条从树中任意节点出发,沿父节点-子节点连接,达到任意节点的序列。同一个节点在一条路径序列中 至多出现一次 。该路径 至少包含一个 节点,且不一定经过根节点。
路径和 是路径中各节点值的总和。
给你一个二叉树的根节点 root ,返回其 最大路径和 。
树的问题往往都是 用递归来解决,这次递归麻烦
如上图, 路径是 15 - 20 - 7 的值最大,不需要打印出路劲节点,只需求最大值,把问题细分下去 对于每一个节点都可以看做是一个根节点
对于一个小树,有几种可能呢,一种是 路劲都走完有最大值,还有一种是 三个节点中取一个节点作为最大值
同理,对于根节点为10 的树(他的全部节点 抽象成左右两个节点),同理也是有两种取法
现在 问题就可以推倒 求一个子树的最大路径值,路劲都是 左中右,中序遍历
递归的参数是 每个叶子节点
递归的出口是 叶子结点为空
递归的逻辑是 左到右
做一个全局变量,用于取最大值
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
| class Solution { public: int maxsum = INT_MIN; int getMax(TreeNode *root){ if(root == nullptr){ return 0; } int lval = max(getMax(root->left),0); int rval = max(getMax(root->right),0); maxsum = max(root->val + lval + rval,maxsum); return root->val + max(lval,rval); } int maxPathSum(TreeNode* root) { getMax(root); return maxsum; } };
|