【深度优先搜索 广度优先搜索】297. 二叉树的序列化与反序列化

本文涉及知识点

深度优先搜索 广度优先搜索
深度优先搜索汇总
图论知识汇总

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

序列化是将一个数据结构或者对象转换为连续的比特位的操作,进而可以将转换后的数据存储在一个文件或者内存中,同时也可以通过网络传输到另一个计算机环境,采取相反方式重构得到原数据。
请设计一个算法来实现二叉树的序列化与反序列化。这里不限定你的序列 / 反序列化算法执行逻辑,你只需要保证一个二叉树可以被序列化为一个字符串并且将这个字符串反序列化为原始的树结构。
提示: 输入输出格式与 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

广度优先搜索

用深度优先搜索也可以,只是麻烦得多。
按树的层次存取,同层次年长的在前。为了不处理负数,保存时将数据加上1000。读取后,减去1000。

保存(序列化)

que记录待处理的节点。vector v记录各节点值对应的字符串,空节点对应空字符串。任意节点只有0个或1个父节点,所以无需visit数组,避免重复处理。
初始根节点入队,按顺序从que取节点,如果为空,则v追加空串。
否则:root的值转成字符串追加到v中。左右子节点入队。
v转成成字符串,中间用逗号隔开。

读取(反序列化)

如果字符串为空,返回null。
建立根节点,入队。
依次出队,读取数据。如果数据为空。忽略。
数据不为空,读取数据到节点,并为当前节点建立左右子节点,左右子节点入队。

代码

class Codec {
public:
	string serialize(TreeNode* root) {
		vector<string> v;
		queue<TreeNode*> que;
		que.emplace(root);
		while (que.size()) {
			auto cur = que.front();
			que.pop();
			if (nullptr == cur) { v.emplace_back(""); continue; }
			v.emplace_back(std::to_string(cur->val+1000));
			que.emplace(cur->left);
			que.emplace(cur->right);
		}
		string str;
		for (const auto& s : v) {
			str += s;
			str += ',';
		}
		str.pop_back();
		return str;
	}
	TreeNode* deserialize(string data) {
		if ("" == data) { return nullptr; }
		vector<int> nums;
		for (int left = 0, r = 0; left < data.length(); left = r) {
			int num = 0;
			while ((r < data.length()) && (',' != data[r])) {
				num = num * 10 + data[r] - '0';
				r++;
			}
			if (left == r) {
				nums.emplace_back(INT_MIN);
			}
			else {
				nums.emplace_back(num-1000);
			}
			r++;
		}
		if (',' == data.back()) {
			nums.emplace_back(INT_MIN);
		}
		TreeNode* root = new TreeNode(nums[0]);
		vector< TreeNode*> nodes;
		nodes.emplace_back(root);
		for (int i = 1; i < nums.size(); i++) {
			if (INT_MIN == nums[i]) {
				continue;
			}
			TreeNode* cur = new TreeNode(nums[i]);
			nodes.emplace_back(cur);
			const int inx = (i - 1) / 2;
			if (1 & i) {
				nodes[inx]->left = cur;
			}
			else {
				nodes[inx]->right = cur;
			}
		}
		return root;
	}
};

2023年6月版,就是用的深度优先

也不是很麻烦。前序遍历,用括号括起来一个子树,可读性似乎好些。

class Codec {
public:

	// Encodes a tree to a single string.
	string serialize(TreeNode* root) {
		auto str = serializeInner(root);
		//std::cout << str << std::endl;
		return str;
	}

	// Decodes your encoded data to tree.
	TreeNode* deserialize(string data) {
		int iPos = 0;
		return deserialize(data, iPos);
	}
private:
	string serializeInner(TreeNode* root) {
		if (nullptr == root)
		{
			return "()";
		}
		return "(" + std::to_string(root->val) + serialize(root->left) + serialize(root->right) + ")";
	}

	TreeNode* deserialize(string data,int& iPos) 
	{
		if (iPos >= data.length())
		{
			return nullptr;
		}
		iPos++;
		if ( ')' == data[iPos])
		{
			iPos ++;
			return nullptr;
		}
		int iValue = 0;
		int iSign = 1;
		if ('-' == data[iPos])
		{
			iSign = -1;
			iPos++;
		}
		while (::isdigit(data[iPos]))
		{
			iValue = iValue * 10 + data[iPos] - '0';
			iPos++;
		}
		iValue *= iSign;
		TreeNode* p = new TreeNode(iValue);
		p->left = deserialize(data, iPos);
		p->right = deserialize(data, iPos);
		iPos++;
		return p;
	}
};

扩展阅读

视频课程

先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。
https://edu.csdn.net/course/detail/38771

如何你想快速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程
https://edu.csdn.net/lecturer/6176

相关推荐

我想对大家说的话
喜缺全书算法册》以原理、正确性证明、总结为主。
按类别查阅鄙人的算法文章,请点击《算法与数据汇总》。
有效学习:明确的目标 及时的反馈 拉伸区(难度合适) 专注
闻缺陷则喜(喜缺)是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。
子墨子言之:事无终始,无务多业。也就是我们常说的专业的人做专业的事。
如果程序是一条龙,那算法就是他的是睛

测试环境

操作系统:win7 开发环境: VS2019 C++17
或者 操作系统:win10 开发环境: VS2022 C++17
如无特殊说明,本算法用**C++**实现。

  • 25
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

闻缺陷则喜何志丹

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值