二叉树层序遍历的几个力扣例题

记录新手小白的做题过程。


题目1:102.二叉树的层序遍历

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

第一次接触层序遍历,还以为很简单,一看题目发现是中等难度的题目。

广度优先搜索

不知道广度优先搜索是什么,所去搜查了一下。

广度优先搜索

所以层序遍历就是将不同的层按顺序放入一个位置,然后依次遍历层数。

官方代码:

class Solution {
public:
    vector<vector<int>> levelOrder(TreeNode* root) {
        vector <vector <int>> ret;//创建vector记录层数中的数,每一层数不止一个,所以ret的元素是rector
        if (!root) {//根结点不为空
            return ret;
        }

        queue <TreeNode*> q;//遍历,一端进一端出保留顺序
        q.push(root);//放入根结点,准备遍历第一层
        while (!q.empty()) {//队列不为空
            int currentLevelSize = q.size();//获取最新一层的元素个数
            ret.push_back(vector <int> ());//想一想,这个用于存放什么?//存放的是最新一层的数据
            for (int i = 1; i <= currentLevelSize; ++i) {//一共取那么多次,都是上一次放进来的那么多个元素,即现在的最新一层
                auto node = q.front(); q.pop();//取出队列第一个,遍历最新一层
                ret.back().push_back(node->val);//放入它的数据//看看这行代码怎么写的,ret的最后一个vector(即在for循环前加入的vector,用于存放最新一层数据).push_back()
                if (node->left) q.push(node->left);//将最新一层的下一层放如队列
                if (node->right) q.push(node->right);//且要注意判断不为空
            }
        }
        
        return ret;
    }
};

看官方解释没有看懂 ,自己琢磨一下,做了个啰里啰唆的注释。

大概就是放入队列的元素是按批次进行的,每次放入一层,在队列取出父结点后,将它不为空的左右子节点进栈,而它自己的数值放入在循环之前创建的vector中

这种存一层,取一层的方法节约了空间

记录自己打代码时犯的错误:

1、在进入将根结点放入队列前,没有判断根结点是否为0,队列的放入是push

2、在for循环前放入vector是这么写的:ret.push_back(vector <int> ());

vector<int>后还有一个括号

3、ret.back().push_back(node->val);//放入元素,不要忘记back是个函数,它的后面有个括号


题目 2.二叉树的层序遍历2

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

思路:掌握了层序遍历后,想要从下至上?Maybe,reverse(ret.begin(),ret.end())从上到下的遍历的数组? 


题目3: 199.二叉树的右视图

在完整的遍历里,我们每过一次循环就要往队列里放一次数据,而依照题目要求,我们只要放这次循环的最后一个数字就可以了。

for(int i=0;i<len;i++){
                TreeNode* p=que.front();
                que.pop();
                if(i==len-1)//每次都取,但只放进去一个
                    ret.push_back(p->val);
                if(p->left) que.push(p->left);
                if(p->right) que.push(p->right);
            }

题目4 :637.二叉树的平均值

这个题目也很好想。将操作改一下就可以了。

while(!que.empty()){
            double len=que.size();
            double sum=0;
            for(int i=0;i<len;i++){
                TreeNode* p=que.front();que.pop();
                sum+=p->val;
                if(p->left) que.push(p->left);
                if(p->right) que.push(p->right);
            }
            ret.push_back(sum/len);
        }

题目5: 429.N叉树的层序遍历

题目:

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

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

 思路:这个题看懵我了,最重要的就是在不确定子结点数量的情况下将子结点全部装入队列中。不过题目已经给出了孩子结点的数组,只要要创建一个循环把子结点装进去就好了。

class Solution {
public:
    vector<vector<int>> levelOrder(Node* root) {
        vector<vector<int>> ret;
        if(root==nullptr) return ret;
        queue<Node*> que;
        que.push(root);
        while(!que.empty()){
            int len=que.size();
            vector<int> cur;
            for(int i=0;i<len;i++){
                Node* p=que.front();
                que.pop();
                cur.push_back(p->val);
                for(auto& ch:p->children){
                    que.push(ch);
                }
            }
            ret.push_back(cur);
        }
        return ret;
    }
};

补充一个知识点: 

关于vector的删除方式


  题目6:515.在每个树行中找最大值

思路:感觉也很简单,换一下循环内的操作就好了。

for(int i=0;i<len;i++){
                TreeNode* p=que.front();que.pop();
                if(p->val>max)
                    max=p->val;
                if(p->left) que.push(p->left);
                if(p->right) que.push(p->right);
            }

 题目7:116.填充每个右节点

题目:

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

struct Node {
  int val;
  Node *left;
  Node *right;
  Node *next;
}
填充它的每个 next 指针,让这个指针指向其下一个右侧节点。如果找不到下一个右侧节点,则将 next 指针设置为 NULL。

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


        while(!que.empty()){
            int len=que.size();
            Node* p;
            for(int i=0;i<len;i++){
                if(i==0){
                    p=que.front();
                    que.pop();
                }
                else{
                    Node* q=que.front();
                    que.pop();
                    p->next=q;
                    p=q;
                }
                if(p->left) que.push(p->left);
                if(p->right) que.push ( p -> right);
            }
        }

注意队列是push,push!它只有一端能进,所以不是push_back

题目8:117.填充每个节点的下一个右侧结点

题目:

给定一个二叉树

struct Node {
  int val;
  Node *left;
  Node *right;
  Node *next;
}
填充它的每个 next 指针,让这个指针指向其下一个右侧节点。如果找不到下一个右侧节点,则将 next 指针设置为 NULL。

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

跟上一题也没差吧?


题目9:104、二叉树的最大深度

题目:

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

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

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

思路:感觉这个就在外圈循环加一个记录的数字就好了。

class Solution {
public:
    int maxDepth(TreeNode* root) {
        if(root==nullptr) return 0;
        queue<TreeNode*> que;
        int i=0;
        que.push(root);
        while(!que.empty()){
            int len=que.size();
            i++;
            for(int j=0;j<len;j++){
                TreeNode* p=que.front();
                que.pop();
                if(p->left) que.push(p->left);
                if(p->right) que.push(p->right);
            }
        }
        return i;
    }
};

 递归法:官方代码

点击链接

class Solution {
public:
    int maxDepth(TreeNode* root) {
        if (root == nullptr) return 0;
        return max(maxDepth(root->left), maxDepth(root->right)) + 1;
    }
};

卧槽,太妙了,每个结点都取它的最长子树距离。因为是从父结点的子结点开始的,所以后面需要加1算上这个递归内的一个父结点 


题目10:111.二叉树的最小深度

题目:

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

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

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

思路:感觉这个emmm,最小深度,是将空也放入队列中,然后像上一题一样记录层数,若遇到空,就直接返回层数。 

上面的思路是错误的,应该是在放结点入队列的最后再判断一下是否两个子结点都为空,都为空说明遇到叶子结点了,就返回层数。

class Solution {
public:
    int minDepth(TreeNode* root) {
        if(root==nullptr) return 0;
        queue<TreeNode*> que;
        int i=0;
        que.push(root);
        while(!que.empty()){
            int len=que.size();
            i++;
            for(int j=0;j<len;j++){
                TreeNode* p=que.front();
                que.pop();
                if(p->left) que.push(p->left);
                if(p->right) que.push(p->right);
                if(p->left==nullptr&&p->right==nullptr)
                    return i;
            }
        }
        return i;
    }
};

好了,题目说难也不难,不过问题是是否能在遇到这一类问题时想起来可以用层序遍历。

下一题! 

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
二叉树的层序遍历是一种广度优先搜索算法,通过逐层地从左到右访问所有节点,返回节点值的顺序。可以使用队列来实现层序遍历。具体思路如下: 1. 创建一个空的结果数组res和一个队列que,将根节点root放入队列que中。 2. 当队列que不为空时,执行以下操作: - 定义一个变量size,用于记录队列que的长度。 - 创建一个空的一维数组vec,用于记录当前层的节点值。 - 遍历0到size-1,执行以下操作: - 弹出队列que的头节点cur。 - 将cur的值cur->val添加到一维数组vec中。 - 若cur的左子节点不为空,则将cur的左子节点压入队列que中。 - 若cur的右子节点不为空,则将cur的右子节点压入队列que中。 - 将一维数组vec添加到结果数组res中。 3. 返回结果数组res作为层序遍历的结果。 以下是一段C++代码示例,实现了二叉树的层序遍历: ``` vector<vector<int>> levelOrder(TreeNode* root) { vector<vector<int>> res; // 结果数组 queue<TreeNode*> que; if (root) que.push(root); while (!que.empty()) { int size = que.size(); // 记录队列的长度 vector<int> vec; for (int i = 0; i < size; i++) { TreeNode* cur = que.front(); que.pop(); vec.push_back(cur->val); if (cur->left) que.push(cur->left); if (cur->right) que.push(cur->right); } res.push_back(vec); } return res; } ``` 这段代码使用了一个队列que,利用队列的先进先出的特点,按照层序遍历的顺序将节点添加到结果数组res中。最后返回结果数组res即可得到二叉树的层序遍历结果。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [力扣刷题 | 二叉树](https://download.csdn.net/download/weixin_38633083/13740772)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] - *2* *3* [力扣:102. 二叉树的层序遍历](https://blog.csdn.net/qq_46111138/article/details/124895548)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值