NC204382 中序序列

题目

题目描述

给定一棵有 n 个结点的二叉树的先序遍历与后序遍历序列,求其中序遍历序列。
若某节点只有一个子结点,则此处将其看作左儿子结点

示例1

输入

5,[3,2,1,4,5],[1,5,4,2,3]

返回值

[1,2,5,4,3]

说明

img

备注

1n100,000

题解

知识点:树,递归。

通常通过先序和后序的组合是无法确定一棵二叉树的,因为根左右和左右根都无法划分左右子树。而本题确定了第一个子节点必然是左孩子,即左子树根节点,那么通过第一个子节点可以确定左右子树。

先通过先序找到根节点和第一个左子树根节点,然后在后序找到左子树根节点位置,其必然在后序左子树区间的最右端。因此,后序的最左端到这个节点的所有节点都是左子树的,而剩下的除了根节点都是右子树的。于是,左右子树划分完成,可以递归实现。

时间复杂度 O(nlogn)

空间复杂度 O(n)

代码

#include <bits/stdc++.h>
using namespace std;
vector<int> fo, lo;
void io(int l1, int r1, int l2, int r2) {
if (l1 > r1) return;
if (l1 == r1) {
cout << fo[l1] << ' ';
return;
}
int pos = -1;
for (int i = l2;i <= r2;i++) if (lo[i] == fo[l1]) pos = i;
io(l1 + 1, l1 + 1 + (pos - l2), l2, pos);
cout << fo[l1] << ' ';
io(l1 + 1 + (pos - l2) + 1, r1, pos + 1, r2 - 1);
}
int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
int n;
cin >> n;
for (int i = 0, tmp;i < n;i++) cin >> tmp, fo.push_back(tmp);
for (int i = 0, tmp;i < n;i++) cin >> tmp, lo.push_back(tmp);
io(0, n - 1, 0, n - 1);
return 0;
}
posted @   空白菌  阅读(37)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
点击右上角即可分享
微信分享提示