【牛客算法】某司面试算法题:循环右移二叉树

一、算法题描述

1.1 算法描述

现有一棵n个节点构成的二叉树,请你将每一层的节点向右循环位移k位。某层向右位移一位(即k=1)的含义为:

  1. 若当前节点为左孩子节点,会变成当前节点的双亲节点右孩子节点

  2. 若当前节点为右儿子,会变成当前节点的双亲节点右边相邻兄弟节点左孩子节点。(如果当前节点的双亲节点已经是最右边的节点了,则会变成双亲节点同级的最左边的节点左孩子节点)

  3. 该层的每一个节点同时进行一次位移。

  4. 是从最下面的层开始位移,位移完每一层之后,再向上,直到根节点,位移完毕。

如果从最后一层开始对该二叉树的每一层循环位移k位。以下方二叉树为例,k=1:

			 1
			 / \
			2   3
			   / \
			  4   5

位移最后一层,5变成2的左孩子节点,4变成3的右孩子节点,如下图:

			    1
			   / \
			  2   3
			 /     \
			5       4

再位移倒数第二层,3变成1的左孩子节点,2变成1的右孩子的节点,它们的孩子节点随着一起位移,如下图:

		      1
		     / \
		    3   2
		    \   /
		     4 5

根节点没有双亲节点,不用位移,位移完毕

现在给你这棵二叉树,请你返回循环右移k位后的二叉树。

1.2 示例

1.2.1 示例1

输入

{1,2,3,#,#,4,5},1

输出

{1,3,2,#,4,5}

说明

解释见题面描述。     

1.2.1 示例2

输入

{1,2,3,4},2

输出

{1,2,3,#,#,4}

说明

      1
     / \
    2   3
   /
  4



	变为



      1
     / \
    2   3
       /
      4

1.2.3 示例3

输入

{1,#,3,4,5},1

输出

{1,3,#,5,4}

说明

    1
     \
      3
     / \
    4   5


变为


    1
     \
      3
     / \
    5   4


变为

        1
       /
      3
     / \
    5   4


1.4 备注

树的节点个数在[1,10^5]之间,且保证该树上每个节点的编号不同,节点编号并非按顺序排列,1≤k≤100

1.5 提供的代码

import java.util.*;

/*
 * public class TreeNode {
 *   int val = 0;
 *   TreeNode left = null;
 *   TreeNode right = null;
 *   public TreeNode(int val) {
 *     this.val = val;
 *   }
 * }
 */

public class Solution {
    /**
     * 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可
     *
     * 
     * @param root TreeNode类 
     * @param k int整型 
     * @return TreeNode类
     */
    public TreeNode cyclicShiftTree (TreeNode root, int k) {
        // write code here
    }
}

二、算法实现

2.1 算法思路

  1. 层序遍历:使用队列进行BFS(层次遍历),同时记录每个节点在每一层的位置信息,包括父节点位置、当前节点位置以及是否为左孩子或右孩子。将这些信息存储在levels数组中,每个元素是一个列表,包含该层所有节点的信息。

  2. 位移操作:从最底层开始,逐层向上进行位移操作。对于每一层,首先计算出这一层需要位移的步数,然后根据位移步数调整每个节点的左右子节点指针。

  3. 指针重构:在位移过程中,需要重新构建每个节点的左右子节点指针。这可以通过遍历levels数组中的每层节点,并根据它们的父节点位置和位移步数来确定新的子节点位置。

  4. 更新指针:在位移操作完成后,需要更新上一层的节点指针,将当前层的节点连接到上一层的相应节点上。

  5. 处理边界条件:在位移过程中,需要注意处理边界条件,例如当位移步数超过当前层节点数时,需要对步数进行取模操作。

2.2 算法实现

import java.util.*;

/*
 * TreeNode类定义
 * public class TreeNode {
 *   int val = 0;
 *   TreeNode left = null;
 *   TreeNode right = null;
 *   public TreeNode(int val) {
 *     this.val = val;
 *   }
 * }
 */

public class Solution {
    
    // Node类封装了节点的层次信息,便于后续重构指针
    class Node {
        int parentIndex;    // 父节点在当前层的位置
        int currentIndex;   // 当前节点在该层的索引位置
        int flag;           // 标记当前节点是左孩子(0)还是右孩子(1)
        TreeNode currentNode;  // 当前节点的引用
        
        // 构造方法
        public Node(int parentIndex, int currentIndex, int flag, TreeNode currentNode) {
            this.parentIndex = parentIndex;
            this.currentIndex = currentIndex;
            this.flag = flag;
            this.currentNode = currentNode;
        }
    }
    
    public TreeNode cyclicShiftTree(TreeNode root, int k) {
        if (root == null) return null;  // 空树处理
        
        // 存储每层的节点信息,用于后续循环位移和指针重构
        ArrayList<ArrayList<Node>> levels = new ArrayList<>();
        Queue<Node> queue = new LinkedList<>();
        
        // 初始化:根节点加入队列
        queue.offer(new Node(0, 0, 0, root));
        
        // `parentNum`记录当前层节点数,`nextNum`记录下一层节点数
        int parentNum = 1, nextNum = 0;
        ArrayList<Node> tempLevel = new ArrayList<>();  // 当前层的节点列表

        // 层次遍历:记录每层节点的结构信息
        while (!queue.isEmpty()) {
            Node node = queue.poll();
            tempLevel.add(node);  // 将节点加入当前层的列表
            --parentNum;

            // 添加左子节点到队列
            if (node.currentNode.left != null) {
                queue.offer(new Node(node.currentIndex, nextNum++, 0, node.currentNode.left));
            }
            // 添加右子节点到队列
            if (node.currentNode.right != null) {
                queue.offer(new Node(node.currentIndex, nextNum++, 1, node.currentNode.right));
            }

            // 当前层遍历完毕,将其加入到levels,并更新下一层
            if (parentNum == 0) {
                parentNum = nextNum;
                nextNum = 0;
                levels.add(tempLevel);
                tempLevel = new ArrayList<>();
            }
        }

        // 从树的最底层开始,逐层循环右移并更新左右子节点指针
        int depth = levels.size() - 1;
        for (int i = depth; i >= 1; --i) {  // 从最底层向上逐层处理
            ArrayList<Node> parentLevel = levels.get(i - 1);
            int parentLevelSize = parentLevel.size();

            // 清空上一层每个节点的左右子节点,准备重新分配指针
            for (Node parent : parentLevel) {
                parent.currentNode.left = null;
                parent.currentNode.right = null;
            }

            // 计算当前层的位移步数
            int move = k % (2 * parentLevelSize);  // 控制位移在有效范围内
            tempLevel = levels.get(i);

            // 根据位移更新每个节点的左右子节点指针
            for (Node node : tempLevel) {
                // 计算目标父节点位置和孩子标记(左孩子或右孩子)
                int targetParentIndex = node.flag == 0
                    ? (node.parentIndex + move / 2) % parentLevelSize
                    : (node.parentIndex + (move + 1) / 2) % parentLevelSize;
                
                int targetChildFlag = node.flag == 0 ? move % 2 : (move + 1) % 2;
                
                // 获取目标父节点
                Node targetParent = parentLevel.get(targetParentIndex);

                // 根据targetChildFlag更新子节点指针
                if (targetChildFlag == 0) {
                    targetParent.currentNode.left = node.currentNode;
                } else {
                    targetParent.currentNode.right = node.currentNode;
                }
            }
        }

        return root;  // 返回根节点,树的结构已完成位移并更新
    }
}

2.3 算法复杂度分析

  1. 时间复杂度:算法的时间复杂度主要取决于树的节点数n以及位移步数k。在最坏情况下,我们需要对每个节点进行位移操作,因此时间复杂度为O(n * k)。然而,由于位移操作的步数k通常远小于树的节点数n,因此实际的时间复杂度通常接近O(n)。

  2. 空间复杂度:算法的空间复杂度主要取决于树的高度h和每层的节点数。在最坏情况下,我们可能需要存储所有层的节点信息,因此空间复杂度为O(n),其中n为树的节点数。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/905601.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

MATLAB发票识别系统

课题介绍 该课题为基于MATLAB的发票识别系统。主要识别发票的编号。可定做发票的日期&#xff0c;金额等字段的识别。通过输入图片&#xff0c;校正&#xff0c;定位目标区域&#xff0c;分割&#xff0c;字符分割&#xff0c;模板匹配识别&#xff0c;得出结果。整个设计包含…

前端拖拽库方案之react-beautiful-dnd

近期&#xff0c;知名 React 拖拽库 react-beautiful-dnd 宣布了项目弃用的决定&#xff0c;未来将不再维护。这一决定源于其存在的缺陷与局限性&#xff0c;促使作者转向开发一个更加现代化的拖拽解决方案——Pragmatic drag and drop&#xff08;下面会介绍&#xff09;&…

Rust 力扣 - 643. 子数组最大平均数 I

文章目录 题目描述题解思路题解代码题解链接 题目描述 题解思路 我们遍历长度为k的窗口&#xff0c;我们只需要记录窗口内的最大和即可&#xff0c;遍历过程中刷新最大值 结果为窗口长度为k的最大和 除以 k 题解代码 impl Solution {pub fn find_max_average(nums: Vec<…

Linux——五种IO模型

目录 一IO基本理解 二五种IO模型 1五种IO模型示意图 2同步IO和异步IO 二非阻塞IO 1fcntl 2实现非阻塞IO 三多路复用 1select 1.1定位和作用 1.2介绍参数 1.3编写多路复用代码 1.4优缺点 2poll 2.1作用和定位 2.2介绍参数 2.3修改select代码 3epoll 3.1介绍…

php解密,sg11解密-sg15解密 如何由sourceGuardian11-sourceGuardian15加密(sg11加密~sg15加密)的源码

sg11加密~sg11加密的PHP文件运行需安装SG11加密-SG15加密组件使用、支持WINDOW及LINUX各版本 sg11解密(SourceGuardian)-sg15解密(SourceGuardian)&#xff0c;号称目前最安全的组件加密已可以解密&#xff0c;解密率99.9%&#xff0c;基本可以直接使用&#xff0c;代码特征是…

Jetson OrinNX平台CSI相机导致cpu load average升高问题调试

1. 前言 硬件: Orin NX JP: 5.1.2, R35.4.1 用v4l2-ctl --stream-mmap -d0 命令去获取相机数据时, 用top查看cpu使用情况, CPU占用率很低,但load average在1左右, 无任何程序运行时,load average 为0 用ps -aux 查看当前进程情况,发现有两个系统进程vi-output, …

qt QIcon详解

1、概述 QIcon是Qt框架中的一个类&#xff0c;专门用于处理和显示图标。它提供了灵活的接口&#xff0c;支持从多种来源加载图标&#xff0c;如文件、资源或系统主题&#xff0c;并且支持多种图像格式&#xff0c;如PNG、JPEG、SVG等。QIcon类能够与Qt中的各种控件&#xff08…

【操作系统实验课】Linux操作基础

1. 打开Ubuntu Ubuntu-22.04 虚拟机安装-CSDN博客 打开虚拟机软件 启动其中的Ubuntu22.04 打开Ubuntu系统终端 2. 创建目录和文件 创建test3目录&#xff1a; 在终端中输入命令&#xff1a;mkdir /test3。此命令用于在根目录下创建test3目录。&#xff08;注意在命令中&…

【Linux-进程间通信】匿名管道的应用-进程池实现+命名管道的应用ClientServer通信

匿名管道的应用--进程池/C实现 当系统中需要处理众多任务时&#xff0c;可以将这些任务分配给多个子进程来分担工作。然而&#xff0c;频繁地创建和销毁进程会导致较高的时间成本。为减少这种开销&#xff0c;可以采取预先创建一组子进程的策略&#xff08;以避免在任务分配时…

java设计模式之创建者模式(5种)

设计模式 软件设计模式&#xff0c;又称为设计模式&#xff0c;是一套被反复利用&#xff0c;代码设计经验的总结&#xff0c;他是在软件设计过程中的一些不断发生的问题&#xff0c;以及该问题的解决方案。 **创建者模式又分为以下五个模式&#xff1a;**用来描述怎么“将对象…

数据库->数据库约束

目录 一、数据库约束 1.定义 2.约束类型 3.NOT NULL 非空约束 4. UNIQUE 唯一约束 5.PRIMARY KEY 主键约束 1.主键的使用 2.把表中的主键交给数据库自己维护 2.1主键列设置为null 则使用自增 2.2插入除了主键以外的所有非空列&#xff08;推荐方法&#xff09; 2.3自…

ValueError: Object arrays cannot be loaded when allow_pickle=False

文章目录 问题解决方法1&#xff1a;allow_pickleTrue解决方法2&#xff1a;降低numpy版本错误原因&#xff1a;python和numpy版本不兼容 问题 Traceback (most recent call last): File “D:\project\test_st\retrieval\read_npy.py”, line 4, in data np.load(‘mosi0__le…

HTML CSS

目录 1. 什么是HTML 2. 什么是CSS ? 3. 基础标签 & 样式 3.1 新浪新闻-标题实现 3.1.1 标题排版 3.1.1.1 分析 3.1.1.2 标签 3.1.1.3 实现 3.1.2 标题样式 3.1.2.1 CSS引入方式 3.1.2.2 颜色表示 3.1.2.3 标题字体颜色 3.1.2.4 CSS选择器 3.1.2.5 发布时间字…

Prometheus和Grafana的安装部署

初识Prometheus和Grafana 通常来说&#xff0c;对于一个运行时的复杂系统&#xff0c;如果系统出了问题是很难排查的。因为你是不太可能在运行时一边检查代码一边调试的。因此&#xff0c;你需要在各种关键点加上监控&#xff0c;通过监控获取的数据&#xff0c;指导我们进一步…

ubuntu20.04 加固方案-设置用户缺省UMASK

一、编辑/etc/profile配置文件 打开终端。 查看当前umask 使用文本编辑器&#xff08;如vim&#xff09;编辑/etc/profile文件。 sudo vim /etc/profile 二、添加配置参数 在打开的配置文件的末尾中&#xff0c;添加或修改以下参数&#xff1a; umask 027 三、保存并退出…

liunx网络套接字 | 实现基于tcp协议的echo服务

前言&#xff1a;本节讲述linux网络下的tcp协议套接字相关内容。博主以实现tcp服务为主线&#xff0c;穿插一些小知识点。以先粗略实现&#xff0c;后精雕细琢为思路讲述实现服务的过程。下面开始我们的学习吧。 ps&#xff1a;本节内容建议了解网络端口号的友友们观看哦。 目录…

【果蔬识别】Python+卷积神经网络算法+深度学习+人工智能+机器学习+TensorFlow+计算机课设项目+算法模型

一、介绍 果蔬识别系统&#xff0c;本系统使用Python作为主要开发语言&#xff0c;通过收集了12种常见的水果和蔬菜&#xff08;‘土豆’, ‘圣女果’, ‘大白菜’, ‘大葱’, ‘梨’, ‘胡萝卜’, ‘芒果’, ‘苹果’, ‘西红柿’, ‘韭菜’, ‘香蕉’, ‘黄瓜’&#xff09;…

isp框架代码理解

一、整体框架如下&#xff1a; 1 外层的src中 1.1 从camera.c->task.c&#xff1a;封装了3层&#xff0c;透传到某个功能的本级。 1.2 core.c和capability.c中实现&#xff1a;开机初始化加载参数。2. plat/src中 2.1 fun.c中继task.c又透传了一层&#xff1b;以及最后功能…

VuePress文档初始化请求过多问题探讨

1. 背景 公司有部门使用VuePress 1.0搭建平台的帮助文档&#xff0c;前期文档不是很多&#xff0c;所以没有暴露出特别明显的问题。但随着文档的逐步迭代和内容的增多&#xff0c;出现了大量的并发请求&#xff0c;总共有218个请求&#xff0c;导致服务器带宽耗尽、响应速度下降…

Paimon x StarRocks 助力喜马拉雅构建实时湖仓

作者&#xff1a;王琛 喜马拉雅数仓专家 小编导读&#xff1a; 本文将介绍喜马拉雅直播的业务现状及数据仓库架构的迭代升级&#xff0c;重点分享基于 Flink Paimon StarRocks 实现实时湖仓的架构及其成效。我们通过分钟级别的收入监控、实时榜单生成、流量监测和盈亏预警&am…