leetcode刷题(剑指offer) 297.二叉树的序列化和反序列化

297.二叉树的序列化与反序列化

序列化是将一个数据结构或者对象转换为连续的比特位的操作,进而可以将转换后的数据存储在一个文件或者内存中,同时也可以通过网络传输到另一个计算机环境,采取相反方式重构得到原数据。

请设计一个算法来实现二叉树的序列化与反序列化。这里不限定你的序列 / 反序列化算法执行逻辑,你只需要保证一个二叉树可以被序列化为一个字符串并且将这个字符串反序列化为原始的树结构。

提示: 输入输出格式与 LeetCode 目前使用的方式一致,详情请参阅 LeetCode 序列化二叉树的格式。你并非必须采取这种方式,你也可以采用其他的方法解决这个问题。

示例 1:

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

示例 2:

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

示例 3:

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

示例 4:

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

提示:

  • 树中结点数在范围 [0, 104]
  • -1000 <= Node.val <= 1000

题解

这道题涉及到了二叉树的序列化与反序列化,由于序列化和反序列化的操作必须是可逆的,因此必须将完整的二叉树序列化(就是将叶子节点下面的节点也都序列化出来,序列化成两个null),这样反序列化之后的树才能是唯一的。以示例一来举例。

将示例一的输入序列化之后得到的结果是这样的。

1,2,3,null,null,4,5,null,null,null,null

可以使用BFS的思想来层序遍历取出各个元素来生成这个字符串,这部分的思路类似于这题,二叉树的层序遍历。

代码实现如下:

public static String serialize(TreeNode root) {if (root == null) {return "";}Queue<TreeNode> queue = new ArrayDeque<>();queue.add(root);StringBuilder sb = new StringBuilder();while (!queue.isEmpty()) {TreeNode node = queue.poll();if (node.val == NULL) {sb.append("null");sb.append(",");continue;}else {sb.append(node.val);sb.append(",");}if (node.left != null) {queue.add(node.left);}else {queue.add(new TreeNode(NULL));}if (node.right != null) {queue.add(node.right);}else {queue.add(new TreeNode(NULL));}}return sb.substring(0, sb.length() - 1);
}

序列化完成之后,就可以考虑反序列化的问题了,我们先将序列化之后的String按照,分割成一个String数组,这里的难点在于如何定位到当前节点的值在序列化之后的String中的位置。

感觉有点规律,但是又不知道规律在哪,可以画图,如下所示。

图中-左边的数字表示这个节点的值在序列化之后的数组中的index,右边的数字表示这个节点的值。

通过观察可以发现,每个节点的两个叶子节点的index,可以根据父节点的index计算出来,如下。

leftIndex = 2 * index + 1;
rightIndex = 2 * index + 2;

因此可以按照序列化同样的逻辑来进行反序列化,使用BFS。

代码实现如下:

package com.offer;import com.jvm.jad.T;
import com.offer.leetcode.datastruct.TreeNode;
import com.springAnnotation.pojo.A;
import com.缺失的第一个正数;
import sun.reflect.generics.tree.Tree;import java.util.*;public class _297二叉树的序列化与反序列化 {public static void main(String[] args) {TreeNode root = new TreeNode(1);root.left = new TreeNode(2);root.right = new TreeNode(3);TreeNode r = root.right;r.left = new TreeNode(4);r.right = new TreeNode(5);System.out.println(serialize(root));System.out.println(deserialize(serialize(root)));}private static int NULL = -1001;// Encodes a tree to a single string.public static String serialize(TreeNode root) {if (root == null) {return "";}Queue<TreeNode> queue = new ArrayDeque<>();queue.add(root);StringBuilder sb = new StringBuilder();while (!queue.isEmpty()) {TreeNode node = queue.poll();if (node.val == NULL) {sb.append("null");sb.append(",");continue;}else {sb.append(node.val);sb.append(",");}if (node.left != null) {queue.add(node.left);}else {queue.add(new TreeNode(NULL));}if (node.right != null) {queue.add(node.right);}else {queue.add(new TreeNode(NULL));}}return sb.substring(0, sb.length() - 1);}// Decodes your encoded data to tree.public static TreeNode deserialize(String data) {if ("".equals(data)) {return null;}String[] nodeVals = data.split(",");TreeNode root = new TreeNode(Integer.parseInt(nodeVals[0]));int index = 0;Queue<TreeNode> queue = new ArrayDeque<>();queue.add(root);while (!queue.isEmpty()) {TreeNode node = queue.poll();index++;String val = nodeVals[2 * index + 1];if (!"null".equals(val)) {node.left = new TreeNode(Integer.parseInt(val));queue.add(node.left);}val = nodeVals[2 * index + 2];if (!"null".equals(val)) {node.right = new TreeNode(Integer.parseInt(val));queue.add(node.right);}}return root;}
}

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C#,雅各布斯塔尔—卢卡斯(Jacobsthal Lucas Number)的算法与源代码

1 雅各布斯塔尔序列 雅各布斯塔尔序列是一个与斐波那契序列类似的加法序列&#xff0c;由递归关系JnJn-12Jn-2定义&#xff0c;初始项J00&#xff0c;J11。序列中的一个数字称为雅可布沙尔数。它们是卢卡斯序列Un&#xff08;P&#xff0c;Q&#xff09;的一种特殊类型&#x…

【CSS】文本溢出省略的两种方式

【CSS】文本溢出省略的两种方式 假设有一个卡片组件&#xff0c;组件里的内容有长有短&#xff0c;我们希望在内容很长的时候省略文字&#xff0c;以保证卡片的高度不会过高。 单行文本 要实现在文字超出时不显示超出部分&#xff0c;并用省略号表示还有更多文字&#xff0…

alibabacloud学习笔记05(小滴课堂)

高并发下的微服务存在的问题 高并发下的微服务容错方案 介绍什么是分布式系统的流量防卫兵Sentinel 微服务引入Sentinel和控制台搭建 每个服务都加上这个依赖。 启动方式&#xff1a; 讲解AliababCloud微服务整合Sentinel限流配置实操 我们在order和video模块都加上。 分别启动…

【数据结构】二叉树链式结构的实现

简单不先于复杂&#xff0c;而是在复杂之后。 文章目录 1. 二叉树链式结构的实现1.1 前置说明1.2 二叉树的遍历1.2.1 前序、中序以及后序遍历1.2.2 层序遍历 1.3 节点个数以及高度等1.4 二叉树基础oj练习1.5 二叉树的创建和销毁 1. 二叉树链式结构的实现 1.1 前置说明 在学习二…

【python】python爱心代码

一、实现效果&#xff1a; 欢迎来到英杰社区https://bbs.csdn.net/topics/617804998 二、准备工作 &#xff08;1)、导入必要的模块&#xff1a; 代码首先导入了需要使用的模块&#xff1a;requests、lxml和csv。 import requests from lxml import etree import csv 如果出现…

Autosar 网络管理 NM

为什么要网络管理 Autosar网络管理之所以重要&#xff0c;可以用比较通俗的话来解释&#xff1a; 想象一下汽车是一个庞大的交流团队&#xff0c;每个成员都是一部分的电子控制单元&#xff08;ECU&#xff09;。这些成员之间需要不断地交换信息&#xff0c;就像团队成员需要…

JSR303参数校验-SpringMVC

文章目录 JSR303技术标准简介JSR303标准几个具体实现框架validation-apijakarta.validation-apihibernate-validatorspring-boot-starter-validation Spring Validationjavax.validation.constraints包下提供的注解org.hibernate.validator.constraints包扩展的注解校验注解默认…

基于控制台的购书系统(Java 语言实现)

&#x1f4da;博客主页&#xff1a;爱敲代码的小杨. ✨专栏&#xff1a;《Java SE语法》|《数据结构与算法》 ❤️感谢大家点赞&#x1f44d;&#x1f3fb;收藏⭐评论✍&#x1f3fb;&#xff0c;您的三连就是我持续更新的动力❤️ &#x1f64f;小杨水平有限&#xff0c;欢…

【Springcloud篇】学习笔记九(十五、十六章):Cloud Alibaba介绍、Nacos服务注册、服务配置中心

第十五章_Cloud Alibaba简介 1.出现SpringCloud Alibaba的原因 SpringCloud Netflix项目进入维护模式 技术的发展 2.SpringCloud Alibaba简介 2.1是什么 2.2能干嘛 2.3去哪下 阿里巴巴中文文档下载网站&#xff1a; spring-cloud-alibaba/README-zh.md at 2022.x alibaba…

Leetcode—382. 链表随机节点【中等】(水塘抽样法)

2024每日刷题&#xff08;一零九&#xff09; Leetcode—382. 链表随机节点 算法思想 我们可以在初始化时&#xff0c;用一个数组记录链表中的所有元素&#xff0c;这样随机选择链表的一个节点&#xff0c;就变成在数组中随机选择一个元素 实现代码 /*** Definition for si…

2024美赛B题Searching for Submersibles原创论文完整版

Searching for Submersibles搜索潜水器 2024美赛B题Searching for Submersibles原创论文&#xff08;共38页&#xff09;部分内容&#xff0c;其余见文末&#xff1a; 整体框架&#xff1a; 1.1 问题背景与问题重述 海上游轮迷你潜水艇公司&#xff08;MCMS&#xff09;&…

如何用gpt快速做好数据分析?

由于技术限制&#xff0c;目前InfinitePaper AI仅支持上传1份文件&#xff0c;且大小不超过10M。但是&#xff0c;在强大的代码解释器面前&#xff0c;这都是小问题。我们只需要将可能用到的文件打包成压缩文件上传即可&#xff0c;之后要求GPT直接解压就能正常完成后续需求。 …