深圳幻海软件技术有限公司 欢迎您!

【数据结构】二叉树基础OJ

2023-07-18

目录1.单值二叉树2.检查两颗树是否相同3.对称二叉树4.二叉树的前序遍历5.二叉树的中序遍历6.二叉树的后序遍历7.另一颗树的子树8.二叉树的结构及遍历世界上没有不劳而获的东西!1.单值二叉树链接:力扣代码1:/***Definitionforabinarytreenode.*structTree

目录

1. 单值二叉树

2. 检查两颗树是否相同

3. 对称二叉树

4. 二叉树的前序遍历

5. 二叉树的中序遍历

6. 二叉树的后序遍历

7. 另一颗树的子树

8. 二叉树的结构及遍历


世界上没有不劳而获的东西!


1. 单值二叉树

链接:力扣

代码1

  1. /**
  2. * Definition for a binary tree node.
  3. * struct TreeNode {
  4. * int val;
  5. * struct TreeNode *left;
  6. * struct TreeNode *right;
  7. * };
  8. */
  9. bool isUnivalTree(struct TreeNode* root)
  10. {
  11. if (root == NULL)
  12. {
  13. return true;
  14. }
  15. if (root->left && root->val != root->left->val)
  16. {
  17. return false;
  18. }
  19. if (root->right && root->val != root->right->val)
  20. {
  21. return false;
  22. }
  23. return isUnivalTree(root->left) && isUnivalTree(root->right);
  24. }

思想:(1)【左孩子的返回值是否相等+右孩子的返回值是否相等】当左孩子和根比较不相等或者右孩子和根比较不相等,这个树就是不相等的【返回true的条件比较多,所以,返回不相等的条件少】【代码1】(2)遍历,和根的值进行比较

注意:(1)在OJ题中尽量不要定义静态变量和全局变量

2. 检查两颗树是否相同

链接:力扣

代码

  1. /**
  2. * Definition for a binary tree node.
  3. * struct TreeNode {
  4. * int val;
  5. * struct TreeNode *left;
  6. * struct TreeNode *right;
  7. * };
  8. */
  9. bool isSameTree(struct TreeNode* p, struct TreeNode* q)
  10. {
  11. if (p == NULL && q == NULL)
  12. {
  13. return true;
  14. }
  15. if (p == NULL || q == NULL)//一个为空,一个不为空
  16. {
  17. return false;
  18. }
  19. if (p->val != q->val)
  20. {
  21. return false;
  22. }
  23. return isSameTree(p->left, q->left) && isSameTree(p->right, q->right);
  24. }

思想:递归;分治

3. 对称二叉树

链接:力扣

代码

  1. /**
  2. * Definition for a binary tree node.
  3. * struct TreeNode {
  4. * int val;
  5. * struct TreeNode *left;
  6. * struct TreeNode *right;
  7. * };
  8. */
  9. bool _isSymmetric(struct TreeNode* p, struct TreeNode* q)
  10. {
  11. //
  12. if (p == NULL && q == NULL)
  13. {
  14. return true;
  15. }
  16. //其中一个为空
  17. if (p == NULL || q == NULL)
  18. {
  19. return false;
  20. }
  21. return p->val == q->val && _isSymmetric(p->left, q->right) && _isSymmetric(p->right, q->left);
  22. }
  23. bool isSymmetric(struct TreeNode* root)
  24. {
  25. if (root == NULL)
  26. {
  27. return true;
  28. }
  29. return _isSymmetric(root->left, root->right);
  30. }

思想:首先,判断树的左子树和右子树是否相等;然后,判断左子树的左子树和右子树的右子树以及左子树的右子树和右子树的左子树是否相等,以及重复这个操作

4. 二叉树的前序遍历

链接:力扣

代码

  1. /**
  2. * Definition for a binary tree node.
  3. * struct TreeNode {
  4. * int val;
  5. * struct TreeNode *left;
  6. * struct TreeNode *right;
  7. * };
  8. */
  9. /**
  10. * Note: The returned array must be malloced, assume caller calls free().
  11. //意思是返回值可以被malloc,假设由调用者释放
  12. */
  13. //首先确定二叉树的节点个数
  14. int BinaryTreeSize(struct TreeNode* root)
  15. {
  16. return root == NULL ? 0 : (BinaryTreeSize(root->left)+BinaryTreeSize(root->right) + 1);
  17. }
  18. int* preorder(struct TreeNode* root, int* a, int* pi)
  19. {
  20. if (root == NULL)
  21. {
  22. return;
  23. }
  24. a[*pi] = root->val;
  25. (*pi)++;
  26. preorder(root->left, a, pi);
  27. preorder(root->right, a, pi);
  28. }
  29. int* preorderTraversal(struct TreeNode* root, int* returnSize)
  30. {
  31. //定义的是数组的指针,因为定义的如果是数组,出了这个函数,就被销毁了
  32. int size = BinaryTreeSize(root);//j节点个数
  33. int* a = (int*)malloc(sizeof(int) * size);//定义数组
  34. if (a == NULL)
  35. {
  36. printf("malloc fail\n");
  37. exit(-1);
  38. }
  39. *returnSize = size;
  40. //前序遍历,并把结果,给数组,
  41. int i = 0;
  42. preorder(root,a, &i);
  43. return a;
  44. }

思想:首先,我们可以非常容易地想到前序遍历的,把之前学到过的前序遍历的printf,改成数组的赋值。看题意,我们需要定义一个数组的指针和二叉树的节点个数,然后尽可以调用前序遍历的函数,进行解决。

注意:returnSize 这个代表的是数组的大小,这个函数,相当于输出型参数,returnSize,外面定义了,但是没有值,需要我们对齐进行赋值。【这里应该用指针,因为形参的改变并不影响实参】

5. 二叉树的中序遍历

链接:力扣

代码

  1. /**
  2. * Definition for a binary tree node.
  3. * struct TreeNode {
  4. * int val;
  5. * struct TreeNode *left;
  6. * struct TreeNode *right;
  7. * };
  8. */
  9. /**
  10. * Note: The returned array must be malloced, assume caller calls free().
  11. */
  12. int BinaryTreeSize(struct TreeNode* root)
  13. {
  14. return root == NULL ? 0 : (BinaryTreeSize(root->left) + BinaryTreeSize(root->right) + 1);
  15. }
  16. void inorder(struct TreeNode* root, int* a, int* pi)
  17. {
  18. if (root == NULL)
  19. {
  20. return;
  21. }
  22. inorder(root->left, a, pi);
  23. a[*pi] = root->val;
  24. (*pi)++;
  25. inorder(root->right, a, pi);
  26. }
  27. int* inorderTraversal(struct TreeNode* root, int* returnSize)
  28. {
  29. int size = BinaryTreeSize(root);
  30. *returnSize = size;
  31. int* a = (int*)malloc(sizeof(int) * size);
  32. if (a == NULL)
  33. {
  34. printf("malloc fail\n");
  35. exit(-1);
  36. }
  37. int i = 0;
  38. inorder(root, a, &i);
  39. return a;
  40. }

6. 二叉树的后序遍历

链接:力扣

代码

  1. /**
  2. * Definition for a binary tree node.
  3. * struct TreeNode {
  4. * int val;
  5. * struct TreeNode *left;
  6. * struct TreeNode *right;
  7. * };
  8. */
  9. /**
  10. * Note: The returned array must be malloced, assume caller calls free().
  11. */
  12. int BinaryTreeSize(struct TreeNode* root)
  13. {
  14. return root == NULL ? 0 : (BinaryTreeSize(root->left) + BinaryTreeSize(root->right) + 1);
  15. }
  16. void postorder(struct TreeNode* root, int* a, int* pi)
  17. {
  18. if (root == NULL)
  19. {
  20. return;
  21. }
  22. postorder(root->left, a, pi);
  23. postorder(root->right, a, pi);
  24. a[*pi] = root->val;
  25. (*pi)++;
  26. }
  27. int* postorderTraversal(struct TreeNode* root, int* returnSize)
  28. {
  29. int size = BinaryTreeSize(root);
  30. *returnSize = size;
  31. int* a = (int*)malloc(sizeof(int) * size);
  32. int i = 0;
  33. postorder(root, a, &i);
  34. return a;
  35. }

7. 另一颗树的子树

链接:力扣

代码

  1. /**
  2. * Definition for a binary tree node.
  3. * struct TreeNode {
  4. * int val;
  5. * struct TreeNode *left;
  6. * struct TreeNode *right;
  7. * };
  8. */
  9. bool isSameTree(struct TreeNode* p,struct TreeNode* q)
  10. {
  11. if (p == NULL && q == NULL)
  12. {
  13. return true;
  14. }
  15. if (p == NULL || q == NULL)
  16. {
  17. return false;
  18. }
  19. if (p->val != q->val)
  20. {
  21. return false;
  22. }
  23. return isSameTree(p->left, q->left) && isSameTree(p->right, q->right);
  24. }
  25. bool isSubtree(struct TreeNode* root, struct TreeNode* subRoot)
  26. {
  27. if (root == NULL)
  28. {
  29. return false;
  30. }
  31. return isSameTree(root, subRoot) || isSubtree(root->left, subRoot) || isSubtree(root->right, subRoot);
  32. }

思想:分治,根中是否有和subRoot相等+左子树中是否有和subRoot相等的部分+左子树中是否有和subRoot相等的部分。

注意:主函数中,我们可能会漏掉if这个条件语句,如果不写这个代码,root->left,就会出现问题。

8. 二叉树的结构及遍历

链接:牛客

代码

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. typedef struct BinaryTreeNode
  4. {
  5. char data;
  6. struct BinaryTreeNode* left;
  7. struct BinaryTreeNode* right;
  8. }BTNode;
  9. //先构建一个二叉树【前序遍历】
  10. BTNode* CreatTree(char* a, int* pi)
  11. {
  12. if (a[*pi] == '#')
  13. {
  14. (*pi)++;
  15. return NULL;
  16. }
  17. //先构建根
  18. BTNode* root = (BTNode*)malloc(sizeof(BTNode));
  19. root->data = a[*pi];
  20. (*pi)++;
  21. //再构建左子树和右子树
  22. root->left = CreatTree(a, pi);
  23. root->right = CreatTree(a, pi);
  24. return root;
  25. }
  26. void InOrder(BTNode* root)
  27. {
  28. if (root == NULL)
  29. {
  30. return;
  31. }
  32. InOrder(root->left);
  33. printf("%c ", root->data);
  34. InOrder(root->right);
  35. }
  36. int main()
  37. {
  38. char a[100];
  39. scanf("%s", a);
  40. int i = 0;
  41. BTNode* tree = CreatTree(a, &i);
  42. InOrder(tree);
  43. free(tree);
  44. tree = NULL;
  45. return 0;
  46. }

思想:先序遍历的思想的字符串,建立二叉树【遇到'#',就返回NULL】,然后再中序遍历的思想进行打印。

 

文章知识点与官方知识档案匹配,可进一步学习相关知识
算法技能树首页概览49291 人正在系统学习中