LeetCode 297. Serialize and Deserialize Binary Tree【树,DFS,BFS,设计,二叉树,字符串】困难

本文属于「征服LeetCode」系列文章之一,这一系列正式开始于2021/08/12。由于LeetCode上部分题目有锁,本系列将至少持续到刷完所有无锁题之日为止;由于LeetCode还在不断地创建新题,本系列的终止日期可能是永远。在这一系列刷题文章中,我不仅会讲解多种解题思路及其优化,还会用多种编程语言实现题解,涉及到通用解法时更将归纳总结出相应的算法模板。

为了方便在PC上运行调试、分享代码文件,我还建立了相关的仓库:https://github.com/memcpy0/LeetCode-Conquest。在这一仓库中,你不仅可以看到LeetCode原题链接、题解代码、题解文章链接、同类题目归纳、通用解法总结等,还可以看到原题出现频率和相关企业等重要信息。如果有其他优选题解,还可以一同分享给他人。

由于本系列文章的内容随时可能发生更新变动,欢迎关注和收藏征服LeetCode系列文章目录一文以作备忘。

0 - 6 个月:字节跳动 3、Facebook 2、亚马逊 2、雅虎 Yahoo 2、特斯拉 2

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

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

提示: 输入输出格式与 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, 10^4] 内
  • -1000 <= Node.val <= 1000

类似题目:

  • 449. 序列化和反序列化二叉搜索树
  • 297. 二叉树的序列化与反序列化 困难
  • 428. 序列化和反序列化 N 叉树 困难

二叉树的序列化本质上是对其值进行编码,更重要的是对其结构进行编码。可以遍历树来完成上述任务。众所周知,我们一般有两个策略:广度优先搜索和深度优先搜索。

下面给出BFS和DFS做法。并借这道题稍微介绍一下拼接字符串的神器,`StringJoiner 类。 S t r i n g J o i n e r StringJoiner StringJoiner 的两种主要用法:

  1. StringJoiner sj = new StringJoiner(",", "[", "]"); 第一个参数表示拼接对象之间的连接符,第二个参数表示拼接后的前缀,第三个参数表示拼接后的后缀。例如将 sj.add("a"); sj.add("b") 之后sj.toString()"[a,b]"
  2. StringJoiner sj = new StringJoiner(","); 相比1,不指定前缀和后缀,上述例子拼接后为 "a,b"

解法1 广度优先搜索

最直接的做法是BFS。对于序列化,通过队列将结点数值依次拼成一个字符串。对当前出队结点 h e a d head head ,考察其左右儿子,若有,则将数字转为字符串后拼接,若无,则拼接 null由于题目并不要求固定的格式,只要我们能从序列化后的字符串反序列化出树即可,因此序列化拼接形式是自由的。可以采用 StringBuilderStringJoiner ,后者内部调用了 StringBuilder ,更便于格式化拼接。「代码」中使用 StringJoiner 完成拼接。

对于反序列化做法类似,也借助队列通过BFS方式完成。先将输入转为数组,利用 idx 跟踪当前反序列化的结点。首节点入队后,进入while循环,结点依次出队,idx 总是依次指向出队 head 结点的左右儿子,若 idx 指向的字符串不为 null ,则将其反序列化为结点,然后将 head 相应儿子指向它。这里用idx < n n n n 是结点字符串数组的大小)来作为while的循环条件。

import java.util.StringJoiner;
public class Codec {public String serialize(TreeNode root) {if (root == null) return "";Queue<TreeNode> q = new ArrayDeque<>();StringJoiner sj = new StringJoiner(",");q.add(root);sj.add(Integer.toString(root.val));while (!q.isEmpty()) {TreeNode head = q.remove();if (head.left != null) {q.add(head.left);sj.add(Integer.toString(head.left.val));} else sj.add("null");if (head.right != null) {q.add(head.right);sj.add(Integer.toString(head.right.val));} else sj.add("null");}return sj.toString();}public TreeNode deserialize(String data) {if (data.length() == 0) return null; // 特判:data == ""String[] nodes = data.split(",");Queue<TreeNode> q = new ArrayDeque<>();TreeNode root = new TreeNode(Integer.parseInt(nodes[0]));q.add(root);int idx = 1, n = nodes.length;while (idx < n) { // 不必以!q.isEmpty()作为判断条件TreeNode head = q.remove();if (!nodes[idx].equals("null")) {TreeNode left = new TreeNode(Integer.parseInt(nodes[idx])); head.left = left; // left挂接到headq.add(left);} idx++;if (!nodes[idx].equals("null")) {TreeNode right = new TreeNode(Integer.parseInt(nodes[idx])); head.right = right; // right挂接到headq.add(right);} idx++;}return root;}
}

解法2 深度优先搜索

广度优先搜索可以按照层次的顺序从上到下遍历所有的节点,深度优先搜索可以从一个根开始,一直延伸到某个叶,然后回到根,到达另一个分支。根据根节点、左节点和右节点之间的相对顺序,可以进一步将深度优先搜索策略区分为:

  • 先序遍历
  • 中序遍历
  • 后序遍历

这里,我们选择先序遍历的编码方式,通过这样一个例子简单理解:

  1. 我们从根节点 1 1 1 开始,序列化字符串是 1 , 1, 1,
  2. 然后我们跳到根节点 2 2 2 的左子树,序列化字符串变成 1 , 2 , 1,2, 1,2, 。现在从节点 2 2 2 开始,我们访问它的左节点 3 3 3 1 , 2 , 3 , N o n e , N o n e , 1,2,3,None,None, 1,2,3,None,None,)和右节点 4 ( 1 , 2 , 3 , N o n e , N o n e , 4 , N o n e , N o n e ) 4(1,2,3,None,None,4,None,None) 4(1,2,3,None,None,4,None,None) N o n e , N o n e , None,None, None,None, 是用来标记缺少左、右子节点,这就是我们在序列化期间保存树结构的方式
  3. 最后,我们回到根节点 1 1 1 并访问它的右子树,它恰好是叶节点 5 5 5 。最后,序列化字符串是 1 , 2 , 3 , N o n e , N o n e , 4 , N o n e , N o n e , 5 , N o n e , N o n e , 1,2,3,None,None,4,None,None,5,None,None, 1,2,3,None,None,4,None,None,5,None,None,

我们可以先序遍历这颗二叉树,遇到空子树的时候序列化成 N o n e None None ,否则继续递归序列化。那么我们如何反序列化呢?首先我们需要根据 , 把原先的序列分割开来得到先序遍历的元素列表,然后从左向右遍历这个序列:

  1. 如果当前的元素为 N o n e None None ,则当前为空树
  2. 否则先解析这棵树的左子树,再解析它的右子树
public class Codec {// Encodes a tree to a single string.public String serialize(TreeNode root) {return serialize(root, new StringBuilder()).toString();}private StringBuilder serialize(TreeNode root, StringBuilder str) {if (root == null) str.append("None,");else {str.append(String.valueOf(root.val) + ",");str = serialize(root.left, str);str = serialize(root.right, str);}return str;}private Integer index;// Decodes your encoded data to tree.public TreeNode deserialize(String data) {String[] tokenArray = data.split(",");index = 0;return deserialize(tokenArray);}public TreeNode deserialize(String[] data) {if (data[index].equals("None")) {++index;return null;}TreeNode root = new TreeNode(Integer.valueOf(data[index]));++index;root.left = deserialize(data);root.right = deserialize(data);return root;}
}

复杂度分析:

  • 时间复杂度:在序列化和反序列化函数中,我们只访问每个节点一次,因此时间复杂度为 O ( n ) O(n) O(n) ,其中 n n n 是节点数,即树的大小。
  • 空间复杂度:在序列化和反序列化函数中,我们递归会使用栈空间,故渐进空间复杂度为 O ( n ) O(n) O(n)

解法3 括号表示编码 + 递归下降解码

我们也可以这样表示一颗二叉树:

  1. 如果当前的树为空,则表示为 X X X
  2. 如果当前的树不为空,则表示为 (<LEFT_SUB_TREE>)CUR_NUM(RIGHT_SUB_TREE) ,其中:
    1. <LEFT_SUB_TREE> 是左子树序列化之后的结果
    2. <RIGHT_SUB_TREE> 是右子树序列化之后的结果
    3. CUR_NUM 是当前节点的值

根据这样的定义,我们很好写出序列化的过程,后序遍历这颗二叉树即可,那如何反序列化呢?根据定义,我们可以推导出这样的巴科斯范式(BNF):
T − > ( T ) n u m ( T ) ∣ X T -> (T) num (T)\ |\ X T>(T)num(T)  X
它的意义是:用 T T T 代表一棵树序列化之后的结果, ∣ | 表示 T T T 的构成为 ( T ) n u m ( T ) (T) num (T) (T)num(T) 或者 X X X ∣ | 左边是对 T T T 的递归定义,右边规定了递归终止的边界条件

因为:

  • T T T 的定义中,序列中的第一个字符要么是 X X X ,要么是 ( ( ( ,所以这个定义是不含左递归的
  • 当我们开始解析一个字符串的时候,如果开头是 X X X我们就知道这一定是解析一个「空树」的结构
  • 如果开头是 ( ,我们就知道需要解析 ( T ) n u m ( T ) (T) num (T) (T)num(T) 的结构,
  • 因此这里两种开头和两种解析方法一一对应,可以确定这是一个无二义性的文法

我们只需要通过开头的第一个字母 X X X 还是 ( ( ( 来判断使用哪一种解析方法。所以这个文法是 L L ( 1 ) LL(1) LL(1) 型文法,如果你不知道什么是 L L ( 1 ) LL(1) LL(1) 型文法也没有关系,只需要知道它定义了一种递归的方法来反序列化,也保证了这个方法的正确性——我们可以设计一个递归函数:

  • 这个递归函数传入两个参数,带解析的字符串和当前当解析的位置 p t r ptr ptr p t r ptr ptr 之前的位置是已经解析的, p t r ptr ptr p t r ptr ptr 后面的字符串是待解析的
  • 如果当前位置为 X X X 说明解析到了一棵空树,直接返回
  • 否则当前位置一定是 ( ( ( ,对括号内部按照 ( T ) n u m ( T ) (T) num (T) (T)num(T) 的模式解析

具体请参考下面的代码。

public class Codec {public String serialize(TreeNode root) {if (root == null) return "X";String left = "(" + serialize(root.left) + ")";String right = "(" + serialize(root.right) + ")";return left + root.val + right;}private int ptr;public TreeNode deserialize(String data) {ptr = 0;return parse(data);}public TreeNode parse(String data) {if (data.charAt(ptr) == 'X') {++ptr;return null;}TreeNode cur = new TreeNode(0);cur.left = parseSubtree(data);cur.val = parseInt(data);cur.right = parseSubtree(data);return cur;}public TreeNode parseSubtree(String data) {++ptr; // 跳过左括号TreeNode subtree = parse(data);++ptr; // 跳过右括号return subtree;}public int parseInt(String data) {int x = 0, sgn = 1;if (!Character.isDigit(data.charAt(ptr))) {sgn = -1;++ptr;}while (Character.isDigit(data.charAt(ptr))) x = x * 10 + data.charAt(ptr++) - '0';return x * sgn;}
}

复杂度分析:

  • 时间复杂度:序列化时做了一次遍历,渐进时间复杂度为 O ( n ) O(n) O(n) 。反序列化时,在解析字符串的时候 p t r ptr ptr 指针对字符串做了一次顺序遍历,字符串长度为 O ( n ) O(n) O(n) ,故这里的渐进时间复杂度为 O ( n ) O(n) O(n)
  • 空间复杂度:考虑递归使用的栈空间的大小,这里栈空间的使用和递归深度有关,递归深度又和二叉树的深度有关,在最差情况下,二叉树退化成一条链,故这里的渐进空间复杂度为 O ( n ) O(n) O(n)

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

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

相关文章

使用动态住宅代理还能带来哪些好处?

一、什么是动态住宅代理ip 动态住宅代理是一种代理技术&#xff0c;它利用代理服务器中转用户和目标服务器之间的网络流量&#xff0c;实现用户真实位置的屏蔽。代理提供商会有自己的ip大池子&#xff0c;当你通过代理服务器向网站发送请求时&#xff0c;服务器会从池子中选中…

JVM快速入门篇

一、JVM探究 1.1前言 请你谈谈你对jvm的理解&#xff1f;Java8虚拟机和之前的变化更新&#xff1f;什么是OOM&#xff1f;什么是栈溢出StackOverFlowError&#xff1f;怎么分析&#xff1f;jvm的常见调优参数有哪些&#xff1f;内存快照如何抓取&#xff1f;怎么分析Dump文件&…

认识模块化

1. 模块化的基本概念 1.1 什么是模块化 模块化是指解决一个复杂问题时&#xff0c;自顶向下逐层把系统划分成若干模块的过程。对于整个系统来说&#xff0c;模块是可组 合、分解和更换的单元。 1. 现实生活中的模块化 2.编程领域中的模块化 编程领域中的模块化&#xff0c;…

【vue3】一些关于hooks的使用经验

前言 最近接到了一个需求&#xff0c;隔壁嵌入式部门希望我们用前端解析渲染Kconfig表单。这篇文章用来记录一下本次使用hook pinia vue3的经验 hooks hooks的概念最早是在 React 中听到的&#xff0c;虽然早些时间也写过一点react&#xff0c;但也只是照葫芦画瓢&#xf…

BT8959T2/5 LCD 驱动

1 Preface/Foreword LCD驱动路径&#xff1a;app/gui/lcd/ SDK默认的LCD驱动&#xff1a;lcd_gc9c01.c 2 数据结构 3 代码分析

更换Eclipse的JDK版本

点击window->Preferences 选择Installed JREs 点击 Add 按钮&#xff0c; 选择Standard VM, 点击 Next。然后选择自己安装的JDK路径

[Vue3 博物馆管理系统] 使用Vue3、Element-plus tabs组件构建选项卡功能

系列文章目录 第一章 定制上中下&#xff08;顶部菜单、底部区域、中间主区域显示&#xff09;三层结构首页 第二章 使用Vue3、Element-plus菜单组件构建菜单 第三章 使用Vue3、Element-plus走马灯组件构建轮播图 第四章 使用Vue3、Element-plus tabs组件构建选项卡功能 [第五…

搭建关键字驱动自动化测试框架

前言 那么这篇文章我们将了解关键字驱动测试又是如何驱动自动化测试完成整个测试过程的。关键字驱动框架是一种功能自动化测试框架&#xff0c;它也被称为表格驱动测试或者基于动作字的测试。关键字驱动的框架的基本工作是将测试用例分成四个不同的部分。首先是测试步骤&#…

教你如何快速阅读葡萄酒标签

我们经常被问及葡萄酒标签上写了什么&#xff0c;总体而言这些信息可以分为四个关键部分&#xff0c;第一品牌或生产商&#xff1b;第二国家或地区&#xff1b;第三葡萄品种&#xff1b;第四年份。 第一品牌或生产商&#xff0c;在寻找葡萄酒的制造商时&#xff0c;著名的品牌名…

论文解读 | 三维点云深度学习的综述

原创 | 文 BFT机器人 KITTI 是作为基准测试是自动驾驶中最具影响力的数据集之一&#xff0c;在学术界和工业界都被广泛使用。现有的三维对象检测器存在着两个限制。第一是现有方法的远程检测能力相对较差。其次&#xff0c;如何充分利用图像中的纹理信息仍然是一个开放性的问题…

DDD 与 CQRS 才是黄金组合

在日常工作中&#xff0c;你是否也遇到过下面几种情况&#xff1a; 使用一个已有接口进行业务开发&#xff0c;上线后出现严重的性能问题&#xff0c;被老板当众质疑&#xff1a;“你为什么不使用缓存接口&#xff0c;这个接口全部走数据库&#xff0c;这怎么能抗住&#xff01…

MATLAB中movmean函数用法

目录 语法 说明 示例 向量的中心移动平均值 向量的尾部移动平均值 矩阵的移动平均值 包含缺失值的向量的移动平均值 基于样本点计算移动平均值 仅返回满窗口平均值 movmean函数的功能是对数据进行移动求平均值。 语法 M movmean(A,k) M movmean(A,[kb kf]) M mov…