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字符串反序列化(动态泛型)

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

哈希leetcode-1

目录 1前言 2.例题  2.1两数之和 2.2判断是否互为字符重排 2.3存在重复元素1 2.4存在重复元素2 2.5字母异位词分组 1前言 哈希表主要是适合于快速查找某个元素(O(1)) 当我们要频繁的查找某个元素,第一哈希表O(1),第二,二分O(log n) 一般可以分为语言自带的容器哈希和用数组模拟的简易哈希。 最简单的比如数组模拟字符存储,只要开26个c

leetcode-24Swap Nodes in Pairs

带头结点。 /*** Definition for singly-linked list.* public class ListNode {* int val;* ListNode next;* ListNode(int x) { val = x; }* }*/public class Solution {public ListNode swapPairs(L

leetcode-23Merge k Sorted Lists

带头结点。 /*** Definition for singly-linked list.* public class ListNode {* int val;* ListNode next;* ListNode(int x) { val = x; }* }*/public class Solution {public ListNode mergeKLists

C++ | Leetcode C++题解之第393题UTF-8编码验证

题目: 题解: class Solution {public:static const int MASK1 = 1 << 7;static const int MASK2 = (1 << 7) + (1 << 6);bool isValid(int num) {return (num & MASK2) == MASK1;}int getBytes(int num) {if ((num &

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟)

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟) 题目描述 给定一个链表,链表中的每个节点代表一个整数。链表中的整数由 0 分隔开,表示不同的区间。链表的开始和结束节点的值都为 0。任务是将每两个相邻的 0 之间的所有节点合并成一个节点,新节点的值为原区间内所有节点值的和。合并后,需要移除所有的 0,并返回修改后的链表头节点。 思路分析 初始化:创建一个虚拟头节点

C语言 | Leetcode C语言题解之第393题UTF-8编码验证

题目: 题解: static const int MASK1 = 1 << 7;static const int MASK2 = (1 << 7) + (1 << 6);bool isValid(int num) {return (num & MASK2) == MASK1;}int getBytes(int num) {if ((num & MASK1) == 0) {return

leetcode105 从前序与中序遍历序列构造二叉树

根据一棵树的前序遍历与中序遍历构造二叉树。 注意: 你可以假设树中没有重复的元素。 例如,给出 前序遍历 preorder = [3,9,20,15,7]中序遍历 inorder = [9,3,15,20,7] 返回如下的二叉树: 3/ \9 20/ \15 7   class Solution {public TreeNode buildTree(int[] pr

【JavaScript】LeetCode:16-20

文章目录 16 无重复字符的最长字串17 找到字符串中所有字母异位词18 和为K的子数组19 滑动窗口最大值20 最小覆盖字串 16 无重复字符的最长字串 滑动窗口 + 哈希表这里用哈希集合Set()实现。左指针i,右指针j,从头遍历数组,若j指针指向的元素不在set中,则加入该元素,否则更新结果res,删除集合中i指针指向的元素,进入下一轮循环。 /*** @param

【每日刷题】Day113

【每日刷题】Day113 🥕个人主页:开敲🍉 🔥所属专栏:每日刷题🍍 🌼文章目录🌼 1. 91. 解码方法 - 力扣(LeetCode) 2. LCR 098. 不同路径 - 力扣(LeetCode) 3. 63. 不同路径 II - 力扣(LeetCode) 1. 91. 解码方法 - 力扣(LeetCode) //思路:动态规划。 cl