• 对称二叉树(Leetcode &101)


    题目

    101. 对称二叉树

    思路

    使用层序遍历,遍历当前层的节点时,如该节点的左(右)孩子为空,在list中添加null,否则加入左(右)孩子的值。每遍历完一层则对当前list进行判断,这里判断我用了一个很笨的方法,前面记录下一层节点值时就设置了两个list,其中一个用来翻转,然后判断这两个list是否相等来判断数是否为对称树。

    去看了解析,有两种方法:递归法、使用双端队列进行迭代。

    代码

    1. public boolean isSymmetric(TreeNode root) {
    2. // 迭代写法:使用双端队列
    3. if(root == null){return true;}
    4. Deque deque = new LinkedList();
    5. deque.offerFirst(root.left);
    6. deque.offerLast(root.right);
    7. while (!deque.isEmpty()){
    8. TreeNode temp_left = deque.pollFirst();
    9. TreeNode temp_right = deque.pollLast();
    10. if(temp_left == null && temp_right == null){continue;}
    11. if(temp_left == null || temp_right == null || temp_left.val != temp_right.val){return false;}
    12. deque.offerFirst(temp_left.right);
    13. deque.offerFirst(temp_left.left);
    14. deque.offerLast(temp_right.left);
    15. deque.offerLast(temp_right.right);
    16. }
    17. return true;
    18. }
    19. public boolean isSymmetric_2(TreeNode root) {
    20. // 递归写法:分解为判断每个子树是否对称
    21. if(root == null){return true;}
    22. return comp(root.left, root.right);
    23. }
    24. public boolean comp(TreeNode left, TreeNode right){
    25. if(left == null && right != null){return false;}
    26. if(left != null && right == null) {return false;}
    27. if(left == null && right == null){return true;}
    28. if(left.val != right.val){return false;}
    29. // 当左右子树都不为空且值相等时,对其左右子树继续进行判断
    30. return comp(left.left, right.right)&&comp(left.right, right.left);
    31. }
    32. public boolean isSymmetric_1(TreeNode root) {
    33. // 判断二叉树是否为轴对称二叉树
    34. // 直接拿层序遍历的结果,看逆转后是否还为原数组来进行判断
    35. if(root == null){return false;}
    36. Queue queue = new ArrayDeque();
    37. queue.add(root);
    38. while (!queue.isEmpty()){
    39. int len = queue.size();
    40. List temp_list = new ArrayList();
    41. List temp_re = new ArrayList();
    42. while (len > 0){
    43. TreeNode temp = queue.poll();
    44. if(temp.left == null){temp_list.add(null);temp_re.add(null);}
    45. if(temp.left != null){queue.add(temp.left);temp_list.add(temp.left.val);temp_re.add(temp.left.val);}
    46. if(temp.right == null){temp_list.add(null);temp_re.add(null);}
    47. if(temp.right != null){queue.add(temp.right);temp_list.add(temp.right.val);temp_re.add(temp.right.val);}
    48. len--;
    49. }
    50. Collections.reverse(temp_list);
    51. if(!temp_list.equals(temp_re)){
    52. return false;
    53. }
    54. }
    55. return true;
    56. }

    题目 

    572. 另一棵树的子树

     思路

    与对称二叉树比较相似,最开始想的是对第一棵树进行层序遍历,找到与子树根节点值一样的节点后,通过同时遍历(这里最开始用的双向队列 但是有问题 后面改成使用两个队列分别存储两棵树的节点情况 循环截至的条件为子树遍历完成)这两棵树来进行判断。 

    后面看了题解,第一种是深度遍历然后匹配,确实层序遍历比较冗余。太久没写也不会写深度遍历了,搞半天前序、中序、后序遍历都是深度遍历!写起来也不是很顺畅,就照着答案敲了一遍,确实比较优雅。

    第二种是将该问题转化为字符串匹配问题,使用KMP(救命,每次学了就忘学了就忘),我只能说泰裤辣!

    代码

    1. // 我的垃圾代码
    2. class Solution {
    3. public boolean isSubtree(TreeNode root, TreeNode subRoot) {
    4. // 层序遍历root子树 找到与subRoot根节点相同的节点
    5. // 队列存储数节点
    6. Queue queue = new LinkedList<>();
    7. queue.add(root);
    8. while(!queue.isEmpty()){
    9. TreeNode node = queue.poll();
    10. if(node.val == subRoot.val){
    11. if(is_same(node, subRoot)){
    12. return true;
    13. }
    14. }
    15. if(node.left != null){queue.add(node.left);}
    16. if(node.right != null){queue.add(node.right);}
    17. }
    18. return false;
    19. }
    20. private boolean is_same(TreeNode root, TreeNode subRoot) {
    21. Queue deque_1 = new LinkedList<>();
    22. Queue deque_2 = new LinkedList<>();
    23. deque_1.add(root);
    24. deque_2.add(subRoot);
    25. while (!deque_2.isEmpty()){
    26. TreeNode node_1 = deque_1.poll();
    27. TreeNode node_2 = deque_2.poll();
    28. if(node_1 == null && node_2 == null){continue;}
    29. if(node_1 == null || node_2 == null || node_1.val != node_2.val){return false;}
    30. deque_1.add(node_1.left);
    31. deque_1.add(node_1.right);
    32. deque_2.add(node_2.left);
    33. deque_2.add(node_2.right);
    34. }
    35. return true;
    36. }
    37. }
    38. // 深度遍历
    39. class Solution {
    40. public boolean isSubtree(TreeNode root, TreeNode subRoot) {
    41. // 题解说深度优先遍历
    42. return dfs(root, subRoot);
    43. }
    44. private boolean dfs(TreeNode root, TreeNode subRoot) {
    45. if (root == null){
    46. return false;
    47. }
    48. return check(root, subRoot) || check(root.left, subRoot) || check(root.right, subRoot);
    49. }
    50. private boolean check(TreeNode root, TreeNode subRoot) {
    51. if(root == null && subRoot == null){return true;}
    52. if(root == null || subRoot == null || root.val != subRoot.val){return false;}
    53. return check(root.left, subRoot.left) && check(root.right, subRoot.right);
    54. }
    55. }

    KMP算法

     

    1. public boolean kmp() {
    2. int sLen = sOrder.size(), tLen = tOrder.size();
    3. int[] next = new int[tOrder.size()];
    4. // 初始化
    5. next[0] = 0;
    6. for (int i = 1, j = 0; i < tLen; i++) {
    7. while (j > 0 && !(tOrder.get(i).equals(tOrder.get(j)))) {
    8. // i != j时,j回退到下标 为next数组前一位的值
    9. j = next[j-1];
    10. }
    11. if (tOrder.get(i).equals(tOrder.get(j))) {
    12. // i == j j++ 更新next数组
    13. j++;
    14. }
    15. next[i] = j;
    16. }
    17. // 根据next数组进行匹配 i指向s j指向模式串t
    18. for (int i = 0, j = 0; i < sLen; i ++) {
    19. while (j > 0 && !(sOrder.get(i).equals(tOrder.get(j)))) {
    20. // 不匹配时 j回退到next数组前一位值的下标
    21. j = next[j-1];
    22. }
    23. if (sOrder.get(i).equals(tOrder.get(j))) {
    24. // 相等j就继续向后走
    25. j++;
    26. }
    27. if (j == tLen) {
    28. // 如果模式串全部都匹配 j会++ 所以这里判断是j == tLen
    29. return true;
    30. }
    31. }
    32. return false;
    33. }

  • 相关阅读:
    yyds,Elasticsearch Template自动化管理新索引创建
    数据结构与算法之排序: 选择排序 (Javascript版)
    Unity中关于多线程的一些事
    了解web框架
    【PinkCAx】可视化工具开发记录与总结
    Javascript文件上传
    力扣——剑指 Offer II 092. 翻转字符
    [LeetCode周赛复盘] 第 305 场周赛20220807
    [重庆思庄每日技术分享]-oracle EM13.4 安装失败之后的处理
    如何测量IIC的建立时间和保持时间
  • 原文地址:https://blog.csdn.net/wangyumei0916/article/details/132699774