【数据结构】--二叉树递归题记

2024-01-12 02:12

本文主要是介绍【数据结构】--二叉树递归题记,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

最近写了几道关于二叉树的剑指offer题,和小伙伴们分享一下心得。

🌈对称的二叉树

请实现一个函数,用来判断一棵二叉树是不是对称的。如果一棵二叉树和它的镜像一样,那么它是对称的。

 思路分析:

对于二叉树的问题来说肯定是用递归进行解决。至于如何解决有待考究,但是

第一步肯定是判断root是不是空。

第二步是进行递归到左右子树进行比较子树的值。这里会遇到两种情况:

  1. 若是都为空那就是对称的返回true.
  2. 若只有一个节点为空或者对称节点间的val值不等,则返回false,因为这两种情况下,该二叉树不满足对称的要求。

第三步:如果对称节点的值相等,则需要进一步比较左子树的左孩子与右子树的右孩子是否对称,以及左子树的右孩子与右子树的左孩子是否对称,若都对称则返回true,否则返回false。

class Solution {
public:bool isSymmetric(TreeNode* root) {// 如果是空树if(!root)//如果结点不是空就进入判断,但是这个逻辑取反,如果是空就进入判断return true;else//进入子节点继续比较return isSymmetric(root->left, root->right);}// 函数重载,此函数比较二叉树中位置对称的两个节点bool isSymmetric(TreeNode* left, TreeNode* right){// 结束条件1:如果对称两个节点都为空,则返回trueif(!left && !right){return true;}// 结束条件2:如果单独一个节点为空,另一个节点不为空,又或者是对称节点间的val值不等,则返回falseif(!left || !right || left->val != right->val)return false;// 结点不是空并且相等return isSymmetric(left->left, right->right) && isSymmetric(left->right, right->left);      }
};

🌈二叉树的镜像 

请完成一个函数,输入一个二叉树,该函数输出它的镜像。

  • 例如输入:

                    4
                   /   \
                 2      7
                / \      / \
              1   3  6   9

  • 镜像输出:

                     4
                    /   \
                 7      2
                 / \      / \
               9   6  3   1

 解题思路:

  • 第一步:根节点是空
  • 第二步:一直递归直到进到最底层根节点位置。从底层开始交换。
  • 第三步:分别记录下最左边,最右边子树的结点值,然后交换,这样层层交换,最后实现镜像。
class Solution {
public:TreeNode* mirrorTree(TreeNode* root) {if (root == nullptr) {return nullptr;}TreeNode* left = mirrorTree(root->left);TreeNode* right = mirrorTree(root->right);root->left = right;root->right = left;return root;}
};

先在栈中存入4,先递归的是左子树,所以把2存入栈中,继续递归到左子树1,但是1是空节点,返回到根2,然后递归2的右子树3,存入栈中,交换1,3后出栈。返回根节点2。

栈底栈底
44
22
1...
3
...

然后递归2的根节点4的右子树,把7压入栈中。同理压入左右子树6,9

栈底栈底
44
22
...7
6
9 ...
交换左右子树后出栈并且返回根节点7。    再交换2,7并且出栈。最后把根节点4出栈    .返回根节点4.         
栈底栈底
44
2...
7...
                                                                       

🌈二叉树剪枝 

给定一个二叉树 根节点 root ,树的每个节点的值要么是 0,要么是 1。请剪除该二叉树中所有节点的值为 0 的子树。

节点 node 的子树为 node 本身,以及所有 node 的后代。

剪枝后的二叉树满足如下要求:

  • 对于每个节点,如果该节点为叶子节点(没有左右子树),且该节点的值为0,则将该节点剪掉。
  • 对于非叶子节点,如果其左子树和右子树都被剪掉了(即为空指针),且该节点的值为0,则将该节点剪掉。

具体实现思路:递归地对每个节点进行处理,首先对其左右子树进行递归,然后判断当前节点是否为叶子节点,如果是则判断其值是否为0,如果是则将该节点剪掉,返回nullptr;如果不是叶子节点,则判断其左右子树是否都为空指针且该节点的值是否为0,如果是则将该节点剪掉,返回nullptr;否则直接返回该节点的指针。

这个最重要的问题就是如何实现把该剪枝的部分置为空?

如果需要被剪枝,则返回nullptr,将当前节点的左子树指针置为nullptr。

class Solution {
public:TreeNode* pruneTree(TreeNode* root) {if (root == nullptr) {return nullptr;}root->left = pruneTree(root->left);//这一句代码的作用是递归调用pruneTree函数来判断当前节点的//左子树是否需要被剪枝,如果需要被剪枝,则返回nullptr,将当前节点的左子树指针置为nullptr,root->right = pruneTree(root->right);if (root->left == nullptr && root->right == nullptr && root->val == 0) {return nullptr;}return root;}
};

你们发现没。这种二叉树的递归问题一般都是从下往上或者从上往下整活。但是从下往上用的比较多虽然效率不高。我们拿第二个举例:

             1

         /       \

      0           1

   /     \       /    \

0      0     0      1

先把1存入栈中,然后递归左子树把0存入栈,在存入0的左子树,

栈底返回值
11
0nullptr
0nullptr

0给根返回空并且出栈,然后调用0的右子树,入栈,它的左右子树都是0,返回nullptr.

同第一左根节点0的左右子树都是0,也进行赋值为nullptr并且出栈。然后1的右子树进栈执行同样操作。

栈底栈底
11
11
01
同理0出栈,1进栈。再返回到第一右子节点1.然后1的左子树赋值为nullptr,最后返回到根节点1.剪枝完成。

这篇关于【数据结构】--二叉树递归题记的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/596475

相关文章

【数据结构】——原来排序算法搞懂这些就行,轻松拿捏

前言:快速排序的实现最重要的是找基准值,下面让我们来了解如何实现找基准值 基准值的注释:在快排的过程中,每一次我们要取一个元素作为枢纽值,以这个数字来将序列划分为两部分。 在此我们采用三数取中法,也就是取左端、中间、右端三个数,然后进行排序,将中间数作为枢纽值。 快速排序实现主框架: //快速排序 void QuickSort(int* arr, int left, int rig

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

《数据结构(C语言版)第二版》第八章-排序(8.3-交换排序、8.4-选择排序)

8.3 交换排序 8.3.1 冒泡排序 【算法特点】 (1) 稳定排序。 (2) 可用于链式存储结构。 (3) 移动记录次数较多,算法平均时间性能比直接插入排序差。当初始记录无序,n较大时, 此算法不宜采用。 #include <stdio.h>#include <stdlib.h>#define MAXSIZE 26typedef int KeyType;typedef char In

leetcode105 从前序与中序遍历序列构造二叉树

根据一棵树的前序遍历与中序遍历构造二叉树。 注意: 你可以假设树中没有重复的元素。 例如,给出 前序遍历 preorder = [3,9,20,15,7]中序遍历 inorder = [9,3,15,20,7] 返回如下的二叉树: 3/ \9 20/ \15 7   class Solution {public TreeNode buildTree(int[] pr

【408数据结构】散列 (哈希)知识点集合复习考点题目

苏泽  “弃工从研”的路上很孤独,于是我记下了些许笔记相伴,希望能够帮助到大家    知识点 1. 散列查找 散列查找是一种高效的查找方法,它通过散列函数将关键字映射到数组的一个位置,从而实现快速查找。这种方法的时间复杂度平均为(

PHP实现二叉树遍历(非递归方式,栈模拟实现)

二叉树定义是这样的:一棵非空的二叉树由根结点及左、右子树这三个基本部分组成,根据节点的访问位置不同有三种遍历方式: ① NLR:前序遍历(PreorderTraversal亦称(先序遍历)) ——访问结点的操作发生在遍历其左右子树之前。 ② LNR:中序遍历(InorderTraversal) ——访问结点的操作发生在遍历其左右子树之中(间)。 ③ LRN:后序遍历(PostorderT

浙大数据结构:树的定义与操作

四种遍历 #include<iostream>#include<queue>using namespace std;typedef struct treenode *BinTree;typedef BinTree position;typedef int ElementType;struct treenode{ElementType data;BinTree left;BinTre

Python 内置的一些数据结构

文章目录 1. 列表 (List)2. 元组 (Tuple)3. 字典 (Dictionary)4. 集合 (Set)5. 字符串 (String) Python 提供了几种内置的数据结构来存储和操作数据,每种都有其独特的特点和用途。下面是一些常用的数据结构及其简要说明: 1. 列表 (List) 列表是一种可变的有序集合,可以存放任意类型的数据。列表中的元素可以通过索

浙大数据结构:04-树7 二叉搜索树的操作集

这道题答案都在PPT上,所以先学会再写的话并不难。 1、BinTree Insert( BinTree BST, ElementType X ) 递归实现,小就进左子树,大就进右子树。 为空就新建结点插入。 BinTree Insert( BinTree BST, ElementType X ){if(!BST){BST=(BinTree)malloc(sizeof(struct TNo

oracle11.2g递归查询(树形结构查询)

转自: 一 二 简单语法介绍 一、树型表结构:节点ID 上级ID 节点名称二、公式: select 节点ID,节点名称,levelfrom 表connect by prior 节点ID=上级节点IDstart with 上级节点ID=节点值 oracle官网解说 开发人员:SQL 递归: 在 Oracle Database 11g 第 2 版中查询层次结构数据的快速