• 【C++】【LeetCode】【二叉树的前序、中序、后序遍历】【递归+非递归】


    目录

    一、二叉树的前序遍历

    递归解法 

    非递归解法

    二、二叉树的中序遍历

    递归解法

    非递归解法

    三、二叉树的后序遍历

    递归解法

    非递归解法


    默认情况下,一个线程的栈空间大小为8MB,当递归的深度太深,我们的程序就容易崩溃

    如果递归的深度太深,栈空间不大,那么程序容易崩溃

    1. #include
    2. using namespace std;
    3. int f(int N)
    4. {
    5. if (N == 1)
    6. return N;
    7. return N + f(N - 1);
    8. }
    9. int main()
    10. {
    11. cout << f(1000000) << endl;
    12. return 0;
    13. }

    改成非递归

    1.改成循环(比方说斐波那契)

    2.通过栈改成循环

     为什么递归会空间溢出,非递归不会空间溢出?

    1.如果是循环的话,形成迭代,消耗的空间很可能是0(1)

    2.如果使用的是通过栈来改成循环的话,我们栈的开辟的空间在内存的堆区

    (内存中栈的空间不大,但是堆的空间很大,几乎不存在空间溢出的问题)

    所以这里我们将递归的遍历改成非递归 

    一、二叉树的前序遍历

    力扣

    递归解法 

    1. /**
    2. * Definition for a binary tree node.
    3. * struct TreeNode {
    4. * int val;
    5. * TreeNode *left;
    6. * TreeNode *right;
    7. * TreeNode() : val(0), left(nullptr), right(nullptr) {}
    8. * TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
    9. * TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
    10. * };
    11. */
    12. class Solution {
    13. public:
    14. void preorderTraversal1(TreeNode* root){
    15. if(root==nullptr)
    16. {
    17. return;
    18. }
    19. tmp.push_back(root->val);
    20. preorderTraversal1(root->left);
    21. preorderTraversal1(root->right);
    22. }
    23. vector<int> preorderTraversal(TreeNode* root) {
    24. if(root==nullptr)
    25. {
    26. return tmp;
    27. }
    28. preorderTraversal1(root);
    29. return tmp;
    30. }
    31. vector<int> tmp;
    32. };

    非递归解法

    二叉树的前序的顺序(根,左子树,右子树)

    1.先访问左路结点

    2.左路结点右子树

     

    我们先访问左子树,然后我们按照顺序将8,3,1依次压入栈中,8位栈底,1为栈顶

    那我们1没有左右子树,直接弹出,然后我们访问到3的时候,我们如何访问3的右树?

    我们如何访问8的,我们就如何访问3

    再拆成左路结点和右路节点,按照我们上面的说明的顺序进行访问。

    1. /**
    2. * Definition for a binary tree node.
    3. * struct TreeNode {
    4. * int val;
    5. * TreeNode *left;
    6. * TreeNode *right;
    7. * TreeNode() : val(0), left(nullptr), right(nullptr) {}
    8. * TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
    9. * TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
    10. * };
    11. */
    12. class Solution {
    13. public:
    14. vector<int> preorderTraversal(TreeNode* root) {
    15. //创建一个辅助栈
    16. stack st;
    17. //创建一个容器v
    18. vector<int> v;
    19. //将cur指向我们的节点
    20. //cur指向结点所代表的树
    21. TreeNode* cur=root;
    22. //cur不为空表示当前cur指向的树还没有访问
    23. //栈不为空表示当前树的右子树还没有访问(栈中的结点的右子树都没有访问)
    24. //所以这两个条件只要有一个没有满足就要继续进行循环
    25. while(cur||!st.empty())
    26. {
    27. //开始访问一棵树
    28. //1.左路结点
    29. while(cur)
    30. {
    31. //将前序遍历的结果压入我们的容器中
    32. v.push_back(cur->val);
    33. //将左路结点放入栈中
    34. st.push(cur);
    35. //不断想左路进行遍历
    36. cur=cur->left;
    37. }
    38. //左路结点的右子树需要访问
    39. //所以我们从栈顶取出我们的栈顶元素,访问其右子树
    40. TreeNode* top=st.top();
    41. st.pop();
    42. cur=top->right;//子问题访问右子树
    43. }
    44. return v;
    45. }
    46. };

    一个结点从栈里面出来,代表着这个结点以及它的左子树访问完了,还剩右节点

    二、二叉树的中序遍历

    力扣

    递归解法

    1. /**
    2. * Definition for a binary tree node.
    3. * struct TreeNode {
    4. * int val;
    5. * TreeNode *left;
    6. * TreeNode *right;
    7. * TreeNode() : val(0), left(nullptr), right(nullptr) {}
    8. * TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
    9. * TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
    10. * };
    11. */
    12. class Solution {
    13. public:
    14. void inorderTraversal1(TreeNode*root){
    15. if(root==nullptr)
    16. {
    17. return;
    18. }
    19. inorderTraversal(root->left);
    20. tmp.push_back(root->val);
    21. inorderTraversal(root->right);
    22. }
    23. vector<int> inorderTraversal(TreeNode* root) {
    24. if(root==nullptr)
    25. {
    26. return tmp;
    27. }
    28. inorderTraversal1(root);
    29. return tmp;
    30. }
    31. vector <int> tmp;
    32. };

    非递归解法

    1. /**
    2. * Definition for a binary tree node.
    3. * struct TreeNode {
    4. * int val;
    5. * TreeNode *left;
    6. * TreeNode *right;
    7. * TreeNode() : val(0), left(nullptr), right(nullptr) {}
    8. * TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
    9. * TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
    10. * };
    11. */
    12. class Solution {
    13. public:
    14. vector<int> inorderTraversal(TreeNode* root) {
    15. //创建一个辅助栈
    16. stack st;
    17. //创建一个容器v
    18. vector<int> v;
    19. //将cur指向我们的根节点
    20. TreeNode* cur=root;
    21. //cur不为空表示当前cur指向的树还没有访问
    22. //栈不为空表示当前树的右子树还没有访问(栈中的结点的右子树都没有访问)
    23. //所以这两个条件只要有一个没有满足就要继续进行循环
    24. while(cur||!st.empty())
    25. {
    26. //开始访问一棵树
    27. //1.左路结点
    28. while(cur)
    29. {
    30. //不断向左路进行遍历
    31. st.push(cur);
    32. cur=cur->left;
    33. }
    34. //左路结点的右子树需要访问
    35. //所以我们从栈顶取出我们的栈顶元素,访问其右子树
    36. TreeNode* top=st.top();
    37. //中序相较于前序就是将结点的数据压入vector的时候放在这个位置
    38. //也就是先将左子树遍历完,全部入栈之后,在出栈的时候,将取出的结点的值追加到我们的vector中
    39. v.push_back(top->val);
    40. st.pop();
    41. cur=top->right;//子问题访问右子树
    42. }
    43. return v;
    44. }
    45. };

     

     前序遍历和中序遍历仅仅是访问根节点的时机不同,代码上也就仅仅是下面的区别

    当左路结点从栈中出来,表示左子树已经访问过了,应该访问这个结点和他的右子树了。 

     

    三、二叉树的后序遍历

    力扣

    递归解法

    1. /**
    2. * Definition for a binary tree node.
    3. * struct TreeNode {
    4. * int val;
    5. * TreeNode *left;
    6. * TreeNode *right;
    7. * TreeNode() : val(0), left(nullptr), right(nullptr) {}
    8. * TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
    9. * TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
    10. * };
    11. */
    12. class Solution {
    13. public:
    14. void postorderTraversal1(TreeNode* root) {
    15. if(root==nullptr)
    16. {
    17. return;
    18. }
    19. postorderTraversal1(root->left);
    20. postorderTraversal1(root->right);
    21. tmp.push_back(root->val);
    22. }
    23. vector<int> postorderTraversal(TreeNode* root) {
    24. if(root==nullptr)
    25. {
    26. return tmp;
    27. }
    28. postorderTraversal1(root);
    29. return tmp;
    30. }
    31. vector<int> tmp;
    32. };

    非递归解法

    后序遍历是先左子树,然后右子树,然后才是根结点

    我们栈中的结点元素仅仅是能保证这个元素的左子树已经被访问过了,所以我们在取出栈顶元素的时候,只有这个结点的右子树也被访问过了,才能从栈中弹出 

     

    比方说我们上面这张图中,依次将8,3,1入栈,1的右子树为空,所以1可以出栈,

    栈变成8->3

    但是哦我们这里的的3,也就是我们的栈顶元素的右子树不是空,并且没有被访问过,所以不能直接弹出,也就是我们的3不能访问

    此时我们将我们的指针指向3的右节点,然后将cur入栈

    此时我们的栈变成了8->3->6->

    然后6有左节点,所以左节点也入栈

    8->3->6->4

    4因为没有右节点,所以4可以出栈,代表我们的4这个结点可以访问了。

    然后我们看此时栈顶的6,6的左已经被访问过了,6的右节点为空,可以弹出了

    然后此时栈顶的元素为3,3的右节点已经被访问过了,所以我们的3也可以被弹出了!也就是我们的3也可以被访问了!

    但是我们这里的3第一次访问的时候右节点没有被访问,我们需要去访问,然后我们第二次去访问3的时候,我们的3的右子树已经被访问过了,我们应该如何区分这两次不同的访问呢? 

    我们可以设置标志位,来标记我们已经访问过这个节点了,但是我们并不知道这样有右子树的结点有多少个,我们如果真的要记录的话,就需要为每一个这样有右树的结点都设置一个标记位。

    也就是可以使用一个map,来标记这个结点,map,来记录访问的次数

    但是我们这样为了一个二叉树还要调用另外一个数据结构,就非常麻烦,有没有别的方法呢? 

    一个结点的右不为空的情况下:

    1.如果右子树没有访问,访问右子树

    2.如果右子树已经访问过了,访问根节点 

    第一次到3的时候,上一个访问的结点是1

    但是我们第二次访问到3的时候,上一个访问的是6

    其实我们可以设置一个前置的标记。

    如果我们的访问的当前结点的前置结点就是我们当前结点的右子树,

    那么救说明我们当前结点的右子树已经被访问过了!

    这样的话,上面的两种情况就可以靠这个前置指针来区分开来了!

    1. /**
    2. * Definition for a binary tree node.
    3. * struct TreeNode {
    4. * int val;
    5. * TreeNode *left;
    6. * TreeNode *right;
    7. * TreeNode() : val(0), left(nullptr), right(nullptr) {}
    8. * TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
    9. * TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
    10. * };
    11. */
    12. class Solution {
    13. public:
    14. vector<int> postorderTraversal(TreeNode* root) {
    15. //创建一个辅助栈
    16. stack st;
    17. //将我们的当前访问的结点的前置结点初始化为空
    18. TreeNode* prev=nullptr;
    19. //创建一个容器v
    20. vector<int> v;
    21. //将cur指向我们的根节点
    22. TreeNode* cur=root;
    23. //cur不为空表示当前cur指向的树还没有访问
    24. //栈不为空表示当前树的右子树还没有访问(栈中的结点的右子树都没有访问)
    25. //所以这两个条件只要有一个没有满足就要继续进行循环
    26. while(cur||!st.empty())
    27. {
    28. //开始访问一棵树
    29. //1.左路结点
    30. while(cur)
    31. {
    32. //将左路结点放入栈中
    33. st.push(cur);
    34. //不断想左路进行遍历
    35. cur=cur->left;
    36. }
    37. //2.左路结点的右子树需要访问
    38. //所以我们从栈顶取出我们的栈顶元素,访问其右子树
    39. TreeNode* top=st.top();
    40. //如果我们没有右子树
    41. //或者我们上一个访问的结点是我们右子树的根(也就是说我们当前结点的右子树已经访问过了)
    42. //按照我们上面的分析,也就是说我们当前的这个结点的左右子树都已经被访问过了,可以将这个点的输入加入我们的容器,然后出栈了
    43. //栈顶结点右子树为空,或者上一节访问结点是有字数的根,说明右子树已经访问过了,可以访问这个栈顶结点
    44. //否则 子问题访问top的右子树
    45. if(top->right==nullptr||top->right==prev)
    46. {
    47. v.push_back(top->val);
    48. //在我们访问下一个界定之前,让我们的前置指针先指向我们当前访问过的元素
    49. prev=top;
    50. cur=nullptr;
    51. st.pop();
    52. }
    53. else
    54. {
    55. //
    56. cur=top->right;//如果右边不为空,就访问右树
    57. }
    58. }
    59. return v;
    60. }
    61. };

     

     

    我们按照上面的说法,先依次将8->3->1入栈,1没有左右子树,直接弹出栈,然后我们当前的prev就是指向1的结点

    然后我们判断3这个结点的有右子树,3有右子树,并且3的前置结点是1结点,并不是我们的6结点,所以我们3的右节点并没有访问过!

    然后我们朝着右子树访问,将6入栈,

    然后再将4入栈

    这时我们的4已经没有左右子树了,我们的4将被弹出栈,然后我们的prev就指向我们的4

    然后对于6,6的右节点为空,所以6即将被弹出,我们的prev就指向我们的6

    然后对于我们的3结点,此刻我们的prev结点指向的是6,也就是我们的3的右节点!!!

    也就是说,我们当前的3结点的右子树已经被访问过了,可以将3弹出了!

    所以我们将3弹出,prev指针指向3

    然后对于8,8有右子树,然后cur指向我们的10

    10没有右子树,然后我们的10就出栈了,

    此时我们的prev指向了10这个结点

    然后我们栈中还剩下8这个结点,此时的prev指针为10,也就是我们8的右子树的结点,所以我们的8这个结点的左右子树都已经完成了访问,所以我们的8也可以出栈了!

     

  • 相关阅读:
    饲料板块毛利润分析主逻辑SQL
    测试工程师应具备何种心态?
    【论文导读】 - 关于联邦图神经网络的3篇文章
    Chrome浏览器是如何工作的?(一)
    dump备份命令
    刷题之Leetcode283题(超级详细)
    栈、队列和数组
    实现免杀:Shellcode的AES和XOR加密策略(vt查杀率:4/70)
    小米云原生文件存储平台化实践:支撑 AI 训练、大模型、容器平台多项业务
    Redis cluster 集群
  • 原文地址:https://blog.csdn.net/weixin_62684026/article/details/127571710