首页 > 编程语言 >【leetcode_C++_二叉树_day12】层序遍历 10 && 226.翻转二叉树&&101. 对称二叉树

【leetcode_C++_二叉树_day12】层序遍历 10 && 226.翻转二叉树&&101. 对称二叉树

时间:2022-11-11 21:46:31浏览次数:44  
标签:node 10 right que 二叉树 && push root 节点

1. 层序遍历

学会二叉树的层序遍历,可以一口气打完以下十题:

  • 102.二叉树的层序遍历
  • 107.二叉树的层次遍历II
  • 199.二叉树的右视图
  • 637.二叉树的层平均值
  • 429.N叉树的层序遍历
  • 515.在每个树行中找最大值
  • 116.填充每个节点的下一个右侧节点指针
  • 117.填充每个节点的下一个右侧节点指针II
  • 104.二叉树的最大深度
  • 111.二叉树的最小深度

队列:先进先出,复合一层一层遍历的逻辑,二叉树广度优先遍历

栈: 先进后出,适合模拟深度优先遍历,也就是递归的逻辑

102.二叉树的层序遍历

给你二叉树的根节点 root ,返回其节点值的 层序遍历 。 (即逐层地,从左到右访问所有节点)。

示例 1:

img

输入:root = [3,9,20,null,null,15,7]
输出:[[3],[9,20],[15,7]]

示例 2:

输入:root = [1]
输出:[[1]]

示例 3:

输入:root = []
输出:[]

102二叉树的层序遍历

代码如下:这份代码也可以作为二叉树层序遍历的模板,打十个就靠它了

class Solution {
public:
    vector<vector<int>> levelOrder(TreeNode* root) {
        queue<TreeNode*> que;//建立一个队列

        if(root!=NULL) que.push(root);
        vector<vector<int>> result;//建立一个二维的数组

        while(!que.empty())
        {
            int size=que.size();//队列的大小
            vector<int> vec;//vec一维数组是用来存储当前高度的节点值
            //这里一定要使用固定大小size,不要使用que.size(),因为que.size是不断变化的
            for(int i=0;i<size;i++)
            {
                TreeNode* node =que.front();
                que.pop();//
                vec.push_back(node->val);//把节点的元素值压入一维数组vec中
                if(node->left) que.push(node->left);//把左孩子压入队列中
                if(node->right) que.push(node->right);//把右孩子压入队列中
            }
            result.push_back(vec);//result是二维数组
        }
        return result;
    }
};

107.二叉树的层次遍历II

给你二叉树的根节点 root ,返回其节点值 自底向上的层序遍历 。 (即按从叶子节点所在层到根节点所在的层,逐层从左向右遍历)

示例 1:

img

输入:root = [3,9,20,null,null,15,7]
输出:[[15,7],[9,20],[3]]

示例 2:

输入:root = [1]
输出:[[1]]

示例 3:

输入:root = []
输出:[]

在模板的基础上reverse函数就可以了

class Solution {
public:
    vector<vector<int>> levelOrderBottom(TreeNode* root) {
        queue<TreeNode*> que;
        if(root!=NULL) que.push(root);
        vector<vector<int>> result;
        while(!que.empty())
        {
            int size = que.size();
            vector<int> vec;
            for(int i=0;i<size;i++)
            {
                TreeNode* node =que.front();
                que.pop();
                vec.push_back(node->val);
                if(node->left) que.push(node->left);
                if(node->right) que.push(node->right);
            }
            result.push_back(vec);
        }
        reverse(result.begin(),result.end());
        return result;
    }
};

199.二叉树的右视图

给定一个二叉树的 根节点 root,想象自己站在它的右侧,按照从顶部到底部的顺序,返回从右侧所能看到的节点值。

示例 1:

img

输入: [1,2,3,null,5,null,4]
输出: [1,3,4]

示例 2:

输入: [1,null,3]
输出: [1,3]

示例 3:

输入: []
输出: []

层序遍历的时候,判断是否遍历到单层的最后面的元素,如果是,就放进result数组中,随后返回result就可以了。

class Solution {
public:
    vector<int> rightSideView(TreeNode* root) {
        queue<TreeNode*> que;
        if(root!=NULL) que.push(root);
        vector<int> result;
        while(!que.empty())
        {
            int size=que.size();
            for(int i=0;i<size;i++)
            {
                TreeNode* node =que.front();
                que.pop();
                //层序遍历的时候,判断是否遍历到单层的最后面的元素,如果是,就放进result数组中,随后返回result就可以了。
                if(i==(size-1)) result.push_back(node->val);
                if(node->left!=NULL) que.push(node->left);
                if(node->right!=NULL) que.push(node->right);
            }
        }
        return result;
    }
};

637.二叉树的层平均值

给定一个非空二叉树的根节点 root , 以数组的形式返回每一层节点的平均值。与实际答案相差 10-5 以内的答案可以被接受。

示例 1:

img

输入:root = [3,9,20,null,null,15,7]
输出:[3.00000,14.50000,11.00000]
解释:第 0 层的平均值为 3,第 1 层的平均值为 14.5,第 2 层的平均值为 11 。
因此返回 [3, 14.5, 11] 。

示例 2:

img

输入:root = [3,9,20,15,7]
输出:[3.00000,14.50000,11.00000]

加入求和语句,最后返回sum/size平均值。注意返回类型是double

class Solution {
public:
    vector<double> averageOfLevels(TreeNode* root) {
        queue <TreeNode*> que;
        if(root!=NULL) que.push(root);
        vector<double> result;
        
        while(!que.empty())
        {
            int size=que.size();
            double sum=0;
            for(int i=0;i<size;i++)
            {
                TreeNode *node=que.front();
                que.pop();
                sum=sum+node->val;
                if(node->left!=NULL)que.push(node->left);
                if(node->right!=NULL) que.push(node->right);
            }
            result.push_back(sum/size);// 将每一层均值放进结果集
        }
        return result;
    }
};

429.N叉树的层序遍历

给定一个 N 叉树,返回其节点值的层序遍历。(即从左到右,逐层遍历)。

树的序列化输入是用层序遍历,每组子节点都由 null 值分隔(参见示例)。

示例 1:

img

输入:root = [1,null,3,2,4,null,5,6]
输出:[[1],[3,2,4],[5,6]]

示例 2:

img

输入:root = [1,null,2,3,4,5,null,null,6,7,null,8,null,9,10,null,null,11,null,12,null,13,null,null,14]
输出:[[1],[2,3,4,5],[6,7,8,9,10],[11,12,13],[14]]

用for循环吧多个节点的孩子push进队列中。

class Solution {
public:
    vector<vector<int>> levelOrder(Node* root) {
        queue<Node*> que;
        if(root!=NULL) que.push(root);
        vector<vector<int>> result;
        while(!que.empty())
        {
            int size=que.size();
            vector<int> vec;
            for(int i=0;i<size;i++)
            {
                Node*node=que.front();
                que.pop();
                vec.push_back(node->val);
                //将节点孩子加入队列
                for(int i=0;i<node->children.size();i++)
                {
                    if(node->children[i]) que.push(node->children[i]);
                }
            }
            result.push_back(vec);
        }      
        return result;
    }
};

515.在每个树行中找最大值

给定一棵二叉树的根节点 root ,请找出该二叉树中每一层的最大值。

示例1:

img

输入: root = [1,3,2,5,3,null,9]
输出: [1,3,9]

示例2:

输入: root = [1,2,3]
输出: [1,3]

设置最大值max,并初始化为INT_MIN. if(node->val>max) max=node->val;

class Solution {
public:
    vector<int> largestValues(TreeNode* root) {
        queue<TreeNode*> que;
        vector<int> result;
        if (root != NULL) que.push(root);
        while(!que.empty())
        {
            int size=que.size();
            int max=INT_MIN;
            for(int i=0;i<size;i++)
            {
                TreeNode*node=que.front();
                que.pop();
                if(node->val>max) max=node->val;
                if(node->left) que.push(node->left);
                if(node->right) que.push(node->right);                
            }
            result.push_back(max);
        }
    return result;
    }
};

116.填充每个节点的下一个右侧节点指针

给定一个 完美二叉树 ,其所有叶子节点都在同一层,每个父节点都有两个子节点。二叉树定义如下:

struct Node {
  int val;
  Node *left;
  Node *right;
  Node *next;
}

填充它的每个 next 指针,让这个指针指向其下一个右侧节点。如果找不到下一个右侧节点,则将 next 指针设置为 NULL

初始状态下,所有 next 指针都被设置为 NULL

示例 1:

img

输入:root = [1,2,3,4,5,6,7]
输出:[1,#,2,3,#,4,5,6,7,#]
解释:给定二叉树如图 A 所示,你的函数应该填充它的每个 next 指针,以指向其下一个右侧节点,如图 B 所示。序列化的输出按层序遍历排列,同一层节点由 next 指针连接,'#' 标志着每一层的结束。

示例 2:

输入:root = []
输出:[]

需要设立头节点和非头节点两种节点。然后把节点连接起来就好了。返回root

class Solution {
public:
    Node* connect(Node* root) {
        queue<Node*> que;
        if(root!=NULL) que.push(root);

        while(!que.empty())
        {
            int size =que.size();
            Node *nodePre;
            Node *node;
            for(int i=0;i<size;i++)
            {
                if(i==0)//头节点
                {
                    nodePre=que.front();
                    que.pop();
                    node=nodePre;
                }
                else
                {
                    node=que.front();
                    que.pop();
                    nodePre->next=node;
                    nodePre=nodePre->next;
                }
                if(node->left) que.push(node->left);
                if(node->right) que.push(node->right);
            }
            nodePre->next=NULL;
        }
        return root;        
    }
};

117.填充每个节点的下一个右侧节点指针II

给定一个二叉树

struct Node {
  int val;
  Node *left;
  Node *right;
  Node *next;
}

填充它的每个 next 指针,让这个指针指向其下一个右侧节点。如果找不到下一个右侧节点,则将 next 指针设置为 NULL

初始状态下,所有 next 指针都被设置为 NULL

进阶:

  • 你只能使用常量级额外空间。
  • 使用递归解题也符合要求,本题中递归程序占用的栈空间不算做额外的空间复杂度。

示例:

img

输入:root = [1,2,3,4,5,null,7]
输出:[1,#,2,3,#,4,5,7,#]
解释:给定二叉树如图 A 所示,你的函数应该填充它的每个 next 指针,以指向其下一个右侧节点,如图 B 所示。序列化输出按层序遍历顺序(由 next 指针连接),'#' 表示每层的末尾。

和上一题的解答其实是一模一样的。

只是上一个是完美二叉树,这个就是普通二叉树罢了。

104.二叉树的最大深度

给定一个二叉树,找出其最大深度。

二叉树的深度为根节点到最远叶子节点的最长路径上的节点数。

说明: 叶子节点是指没有子节点的节点。

示例:
给定二叉树 [3,9,20,null,null,15,7]

    3
   / \
  9  20
    /  \
   15   7

返回它的最大深度 3 。

不需要读取元素值,只需要纪录遍历的次数就好了,遍历的次数就是深度。

class Solution {
public:
    int maxDepth(TreeNode* root) {
        queue<TreeNode*> que;
        if(root!=NULL) que.push(root);
        int depth=0;
        while(!que.empty())
        {
            int size=que.size();
            depth++;//记录深度
            for(int i=0;i<size;i++)
            {
                TreeNode* node=que.front();
                que.pop();
                if(node->left) que.push(node->left);
                if(node->right) que.push(node->right);             
            }
        }
        return depth;
    }
};

111.二叉树的最小深度

给定一个二叉树,找出其最小深度。

最小深度是从根节点到最近叶子节点的最短路径上的节点数量。

说明:叶子节点是指没有子节点的节点。

示例 1:

img

输入:root = [3,9,20,null,null,15,7]
输出:2

示例 2:

输入:root = [2,null,3,null,4,null,5,null,6]
输出:5

左右孩子都为空,就说明到底了,该return 结果了。

/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    int minDepth(TreeNode* root) {
        queue<TreeNode*> que;
        int deep=0;
        if(root!=NULL) que.push(root);
        while(!que.empty())
        {
            int size=que.size();
            deep++; 
            for(int i=0;i<size;i++)
            {
                TreeNode* node=que.front();
                que.pop();
                if(node->left) que.push(node->left);
                if(node->right) que.push(node->right);
                if(node->left==NULL&&node->right==NULL) return deep;
            }                      
        }
        return deep;
    }
};

226.翻转二叉树 (优先掌握递归)

给你一棵二叉树的根节点 root ,翻转这棵二叉树,并返回其根节点。

示例 1:

img

输入:root = [4,2,7,1,3,6,9]
输出:[4,7,2,9,6,3,1]

示例 2:

img

输入:root = [2,1,3]
输出:[2,3,1]

示例 3:

输入:root = []
输出:[]

思路

226.翻转二叉树1

遍历的过程中去翻转每一个节点的左右孩子就可以达到整体翻转的效果。

注意只要把每一个节点的左右孩子翻转一下,就可以达到整体翻转的效果

这道题的难度不高,但是需要想一下不同的遍历方式的应用。相当于对之前的学习内容的总结。

这道题目使用前序遍历和后序遍历都可以,唯独中序遍历不方便(不是不行,是不方便)

image-20221111211423241

class Solution {
public:
//递归前序遍历法:中左右
    TreeNode* invertTree(TreeNode* root) {
        if(root==NULL) return root;
        swap(root->left,root->right);//中
        invertTree(root->left);//左
        invertTree(root->right);//右
        return root;
    }
//迭代前序遍历
    TreeNode* invertTree(TreeNode* root)
    {
        if(root==NULL) return root;
        stack<TreeNode*> st;
        st.push(root);
        while(!st.empty())
        {
            TreeNode *node =st.top();//中
            st.pop();
            swap(node->left,node->right);
            if(node->right) st.push(node->right);//右
            if(node->left) st.push(node->left);//左
        }
        return root;
    }

//迭代统一法前序遍历--迭代法涉及栈,即深度优先遍历
    TreeNode* invertTree(TreeNode* root)
    {
        if(root==NULL) return root;
        stack<TreeNode*> st;
        st.push(root);
        while(!st.empty())
        {
            
            TreeNode *node =st.top();
            if(node!=NULL)
            {
                st.pop();
                if(node->right) st.push(node->right);//右
                if(node->left) st.push(node->left);//左
                st.push(node);//中--即要处理的元素
                st.push(NULL);//在要处理的元素后面加一个NULL
            }
            else
            {
                st.pop();
                node=st.top();
                st.pop();
                swap(node->left,node->right);
            }

        }
        return root;
    }
};

//层序遍历--队列,即广度优先遍历
    TreeNode* invertTree(TreeNode* root)
    {
        queue<TreeNode*> que;
        if(root!=NULL) que.push(root);

        while(!que.empty())
        {
            int size=que.size();
            for(int i=0;i<size;i++)
            {
                TreeNode* node=que.front();
                que.pop();
                swap(node->left,node->right);
                if (node->left) que.push(node->left);
                if (node->right) que.push(node->right);    
            }
        }
        return root;
    }
};

101. 对称二叉树 (优先掌握递归)

给你一个二叉树的根节点 root , 检查它是否轴对称。

示例 1:

img

输入:root = [1,2,2,3,4,4,3]
输出:true

示例 2:

img

输入:root = [1,2,2,null,3,null,3]
输出:false

思路

101. 对称二叉树1

1. 递归法三部曲

标签:node,10,right,que,二叉树,&&,push,root,节点
From: https://www.cnblogs.com/MLcaigou/p/16882109.html

相关文章

  • Win10重装IE
    Win10重装IE卸载IE主要参考:IE浏览器的卸载卸载IE分两步,在系统功能中关闭IE11的服务即可停用IE,但此时IE的文件夹还在,这也就意味着重新启用该服务即可重启IE;若想完全卸载I......
  • 【TS】1103- 30个小知识让你更清楚TypeScript
    TypeScript是Microsoft开发的JavaScript的开源超集,用于在不破坏现有程序的情况下添加附加功能。由于其独特的优势,例如,静态类型和许多速记符号,TypeScript现在被前端和......
  • 【JS】1012- 52个JavaScript常用工具函数整理
    1、isStatic:检测数据是不是除了symbol外的原始数据。functionisStatic(value){return(typeofvalue==='string'||typeofvalue==='number'|......
  • 【Vuejs】1000- 一步一步实现 Vue 3 Reactivity
    Vue3中的响应式原理可谓是非常之重要,通过学习Vue3的响应式原理,不仅能让我们学习到Vue.js的一些设计模式和思想,还能「帮助我们提高项目开发效率和代码调试能力」。在这......
  • 幂次方表达:p1010
    1题目ID:P1010[NOIP1998普及组]幂次方2题目描述:任何一个正整数都可以用 22 的幂次方表示。例如 137=2^7+2^3+2^0137=27+23+20。同时约定方次......
  • 2010 Principles on the Security of AES against First and Second-Order Differenti
    一、对于AES算法的DPA攻击准则(无防护措施下的AES实现)(1)在第3轮列混淆前任意中间值可以用于一阶DPA攻击,该攻击将明文的0,3或15比特固定(2)在第7轮轮......
  • 软件工程 张雯雯 223201062202
    2.1实验过程2.1.1实验代码给出博客园截图和网址,并附上github的代码地址。GitHub代码地址: 2.1.2实验过程(1) 本人角色自己在本次结对项目中所担任角色,以及结对伙......
  • loj #10069. 「一本通 3.1 练习 4」Tree
    给你一个无向带权连通图,每条边是黑色或白色。求一棵最小权的恰好有K条白色边的生成树。题目保证有解  二分一个增加量md, 给每个白边权值加md,跑一下kruskal,......
  • AIR32F103(五) FreeRTOSv202112核心库的集成和示例代码
    目录AIR32F103(一)合宙AIR32F103CBT6开发板上手报告AIR32F103(二)Linux环境和LibOpenCM3项目模板AIR32F103(三)Linux环境基于标准外设库的项目模板AIR32F103(四)2......
  • win10重装如何跳过微软账号直接设置本地帐户
    ​在添加你的帐户界面,选择脱机帐户第二个页面,选择有限的体验第三个页面,设置自己本地的用户名第四个页面,设置自己本地的密码......