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

2024-02-04 00:52

本文主要是介绍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;}
}

这篇关于leetcode刷题(剑指offer) 297.二叉树的序列化和反序列化的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java中JSON格式反序列化为Map且保证存取顺序一致的问题

《Java中JSON格式反序列化为Map且保证存取顺序一致的问题》:本文主要介绍Java中JSON格式反序列化为Map且保证存取顺序一致的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未... 目录背景问题解决方法总结背景做项目涉及两个微服务之间传数据时,需要提供方将Map类型的数据序列化为co

RedisTemplate默认序列化方式显示中文乱码的解决

《RedisTemplate默认序列化方式显示中文乱码的解决》本文主要介绍了SpringDataRedis默认使用JdkSerializationRedisSerializer导致数据乱码,文中通过示... 目录1. 问题原因2. 解决方案3. 配置类示例4. 配置说明5. 使用示例6. 验证存储结果7.

SpringBoot实现Kafka动态反序列化的完整代码

《SpringBoot实现Kafka动态反序列化的完整代码》在分布式系统中,Kafka作为高吞吐量的消息队列,常常需要处理来自不同主题(Topic)的异构数据,不同的业务场景可能要求对同一消费者组内的... 目录引言一、问题背景1.1 动态反序列化的需求1.2 常见问题二、动态反序列化的核心方案2.1 ht

SpringBoot项目中Redis存储Session对象序列化处理

《SpringBoot项目中Redis存储Session对象序列化处理》在SpringBoot项目中使用Redis存储Session时,对象的序列化和反序列化是关键步骤,下面我们就来讲讲如何在Spri... 目录一、为什么需要序列化处理二、Spring Boot 集成 Redis 存储 Session2.1

Java controller接口出入参时间序列化转换操作方法(两种)

《Javacontroller接口出入参时间序列化转换操作方法(两种)》:本文主要介绍Javacontroller接口出入参时间序列化转换操作方法,本文给大家列举两种简单方法,感兴趣的朋友一起看... 目录方式一、使用注解方式二、统一配置场景:在controller编写的接口,在前后端交互过程中一般都会涉及

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

如何配置Spring Boot中的Jackson序列化

《如何配置SpringBoot中的Jackson序列化》在开发基于SpringBoot的应用程序时,Jackson是默认的JSON序列化和反序列化工具,本文将详细介绍如何在SpringBoot中配置... 目录配置Spring Boot中的Jackson序列化1. 为什么需要自定义Jackson配置?2.

Django序列化中SerializerMethodField的使用详解

《Django序列化中SerializerMethodField的使用详解》:本文主要介绍Django序列化中SerializerMethodField的使用,具有很好的参考价值,希望对大家有所帮... 目录SerializerMethodField的基本概念使用SerializerMethodField的

Jackson库进行JSON 序列化时遇到了无限递归(Infinite Recursion)的问题及解决方案

《Jackson库进行JSON序列化时遇到了无限递归(InfiniteRecursion)的问题及解决方案》使用Jackson库进行JSON序列化时遇到了无限递归(InfiniteRecursi... 目录解决方案‌1. 使用 @jsonIgnore 忽略一个方向的引用2. 使用 @JsonManagedR

Java中JSON字符串反序列化(动态泛型)

《Java中JSON字符串反序列化(动态泛型)》文章讨论了在定时任务中使用反射调用目标对象时处理动态参数的问题,通过将方法参数存储为JSON字符串并进行反序列化,可以实现动态调用,然而,这种方式容易导... 需求:定时任务扫描,反射调用目标对象,但是,方法的传参不是固定的。方案一:将方法参数存成jsON字