SCAU 18724 二叉树的遍历运算

2023-11-09

18724 二叉树的遍历运算

Description

二叉树的三种遍历都可以通过递归实现。
如果我们知道一棵二叉树的先序和中序序列,可以用递归的方法求后序遍历序列。

输入格式

两行,第一行一个字符串,表示树的先序遍历,第二行一个字符串,表示树的中序遍历。树的结点一律用小写字母表示。

输出格式

一个字符串,树的后序序列。

输入样例

abcde
bcade

输出样例

cbeda

思路

参考:https://blog.csdn.net/eebaicai/article/details/89788098
因为中序遍历时,根结点总是在其左子树被遍历完之后才被遍历。
因此二叉树的先序/后序遍历可以确定根节点,中序遍历可以区分左右子树。
通过在中序遍历的字符串中找到根节点的位置,位置左边的即为根节点的左子树,右边的为右子树。

因此我们需要通过某一子树(可为整棵树)在前序遍历中的根结点 i,通过在中序遍历中找到该结点的位置 j 分出左右子树,然后推导出 i 的左右子树根结点在前序遍历中的位置,即可不断向下二分。

例:根据先序遍历:abcde a为第一个根节点,则在中序遍历:bcade 中a为根节点的位置,bc为a的左子树,de为a的右子树。
由 前序遍历的结构【根结点 - 左子树 - 右子树】知:
左子树的根结点很容易确认,即前序遍历中根结点的下一个结点 b,而右子树的根结点可,为根结点的位置 + 左子树的长度 + 1 = 0 + 2 + 1 = 3 → d。

定义一个函数 traversal ,传入参数为
①某一子树 在【前序遍历 preorder】中的的根结点 i,
②该子树在【中序遍历 inorder】中开始的位置start,及
③结束的位置end,

第一次执行即为①根结点 0 ,②整棵树的开始位置 0 及③结束位置 inorder.length - 1。

通过在 中序遍历inorder 的位置范围(start~end)内,
找到一个 j 使得 inorder[j] == preorder[i]
即可在中序遍历中划分出根结点 inorder[j] 的左子树: [start, j - 1],
右子树:[j + 1, end],
可知 左子树的长度 leftTreeLength = j - 1 - start + 1 = j - start
左子树的根节点在【先序遍历字符数组preorder】中的位置为 i + 1,
右子树的根节点在【先序遍历字符数组preorder】中的位置,
为 i + leftTreeLength + 1 = i + j - start + 1。

对每一个根节点 preorder[i],两次调用函数 traversal ,分别处理 preorder[i]的左子树(该左子树的根节点为 preorder[i + 1],在 inorder 中开始的位置仍为start,结束的位置为 j - 1),和右子树(根节点为 s1[i + j - start + 1],在inorder 中开始的位置为 j + 1,结束的位置为end)

当start == end,即该子树中只有一个结点(叶子结点),则将该叶子结点存入数组。
对每一个根节点执行完左右子树的递归操作后,将根节点存入数组。(对应后序遍历根节点最后输出)

最后遍历数组输出即可。

代码

#include <iostream>
#include <cstring>
using namespace std;
string preorder,inorder;
char tree[10005] = {0};
int p = 0;
//i 根节点在 preorder 中的位置
//start 子树在 inorder 中开始的位置 
//end 子树在 inorder 中结束的位置
int traversal(int i,int start,int end)
{
     if(start == end)//叶子节点
      {
           tree[p++] = inorder[start];//存入数组
           return 1;
      }
    int j = start;
    //找到 preorder[i](根节点)在 inorder 中的位置
    while(preorder[i] != inorder[j]) j++;
  
    //j-start 左子树的长度   end-j 右子树的长度
    //i+1 左子树的根节点在 preorder 中的位置 i+j-start+1 右子树的根节点在 preorder 中的位置
    if(j != start)//左子树不为空
   		traversal(i+1, start, j-1);//先存储左子树,对应后序遍历左孩子先输出
	if(j != end)//右子树不为空
    	traversal(i+j-start+1, j+1, end);//右子树
    tree[p++] = inorder[j];//根节点最后输出
}

int main()
{
    while(cin>>preorder>>inorder)
    {
        int len = preorder.length();
	    traversal(0,0,len-1);
	    for(int i = 0;i < len;i++)
	   		 cout<<tree[i];
	    cout<<endl;
    }
return 0;
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

SCAU 18724 二叉树的遍历运算 的相关文章

随机推荐

  • python中imread用法_Python matplotlib.pyplot.imread()用法及代码示例

    Matplotlib是Python中的一个库 它是数字的 NumPy库的数学扩展 Pyplot是Matplotlib模块的基于状态的接口 该模块提供了MATLAB like接口 matplotlib pyplot imread 功能 mat
  • Android 11 设置开机默认系统横屏显示

    实现 默认横屏有两套方案 第一种方式 目录 device rockchip rk356x BoardConfig mk SF PRIMARY DISPLAY ORIENTATION 90 第二种方式 Android系统默认是竖屏显示的 想要
  • Git fetch、pull以及merge之间的区别

    笔者当时学习git的时候对fetch以及pull命令之间的区别疑惑不解 被困扰了许久 其实还是对git的原理理解不深才会有这种情况 git对每次提交都会生成一个cmmit id 我们工作区间版本改变其实就是HEAD指针指向的commit i
  • [Git & GitHub] Windows下安装git,从0开始搭建git环境(配置环境变量+设置git-ssh key...配置)(超全版)

    目录 前提准备 安装Git Git配置 配置环境变量 git配置 ssh认证配置过程 配置邮箱和用户名 个人身份 文本换行符配置 前提准备 下载地址 点击此处 点击Windows进行下载 若下载比较慢 点击此处 安装Git 下载之后 双击G
  • Android安全检测-Intent Scheme URLs攻击风险

    一 漏洞原理 利用intent scheme URLs 意图协议URL 可以通过web页面发送intent来启动App应用 攻击者可构造特殊格式的URL直接向系统发送意图 启动App应用的Activity组件或者发送异常数据 导致应用的敏感
  • python3.7添加dlib模块

    1 下载dlib安装包 安装dlib真是费劲 dlib下载地址 http dlib net files 我下载的是dlib 19 14 zip 然后解压安装dlib 在安装dlib前需要安装Boost和Cmake dlib19之后你需要安装
  • Windows DWrite 组件 RCE 漏洞 (CVE-2021-24093) 分析

    聚焦源代码安全 网罗国内外最新资讯 概述 Windows图形组件DWrite库存在数组越界写漏洞 CVE 2021 24093 可导致远程代码执行 当DWrite库解析恶意构造的字体文件时 计算内存分配长度时出现错误 触发越界写 而且字体文
  • simulink电力电子仿真(2)单相桥式半控整流电路实验

    simulink电力电子仿真 2 单相桥式半控整流电路实验 返回目录 主要是赶上了疫情 然后期末要疯狂补实验报告 就索性写一下吧 万一以后再做电力电路仿真 可能会有用的 也希望可以帮助别人 器件的选择及位置 MATLAB的版本 2018a
  • <Linux开发>系统移植 -之- linux构建BusyBox根文件系统及移植过程详细记录

    Linux开发 系统移植 之 linux构建BusyBox根文件系统及移植过程详细记录 前言 本章节讲解的是构建移植BusyBox根文件系统到linux开发板 主要是基于正点原子Linux开发板操作 接下来讲解具体过程记录 BusyBox源
  • sql概念模型和逻辑模型

    一 概念模式和E R图 概念模型的表示方法很多 目前比较常用的是实体联系模型 简称E R模型 E R模型主要用E R图来表示 实体间的联系有 一对一联系 一对多联系 多对多联系 E R模型用矩形框表示现实世界中的实体 用菱形框表示实体间的联
  • Linux开发(七):多线程通信与同步

    线程间无需特别的手段进行通信 因为线程间可以共享数据结构 也就是一个全局变量可以被两个线程同时使用 不过要注意的是线程间需要做好同步 目录 一 互斥锁 1 初始化 1 动态初始化 2 静态初始化 2 加锁 3 解锁 4 销毁互斥锁 5 互斥
  • 为什么bytes32等于uint256

    先说1byte等于8个字节 bytes32则等于8 32 256个字节 接着uint8同样等于8个字节 uint256即8个字节的32倍 256 8 32 因此看到byteX和uintY时 如果X 8 Y 意味着byteX uintY
  • JVM Troubleshooting命令-jinfo

    概述 用来查看正在运行的Java应用程序的扩展参数 支持在运行时 修改部分参数 命令格式 jinfo option pid jinfo option executable core jinfo option servier id remot
  • 代码坏味道与重构之重复代码

    文章目录 1 重复代码的特征 2 重复代码的影响 3 重复代码的重构技巧 1 重复代码的特征 重复代码是代码坏味道的典型代表 重复代码是指相同或相似的代码在一个以上的地方出现 通常有以下几种情形 同一个类 多个方法间重复 子类之间代码重复
  • ChatGPT API 中文版(google翻譯)

    https platform openai com docs api reference introduction 介紹 您可以通過任何語言的 HTTP 請求 我們的官方 Python 綁定 我們的官方 Node js 庫或社區維護的庫與
  • 1566 重复至少 K 次且长度为 M 的模式(模拟)

    1 问题描述 给你一个正整数数组 arr 请你找出一个长度为 m 且在数组中至少重复 k 次的模式 模式 是由一个或多个值组成的子数组 连续的子序列 连续 重复多次但 不重叠 模式由其长度和重复次数定义 如果数组中存在至少重复 k 次且长度
  • 微信小程序的事件绑定、接收参数、示例

    1 微信小程序的事件类别 tap 点击事件 input 输入事件 longtap 长按事件 touchstart 触摸开始 touchend 触摸结束 touchcansce 取消触摸 注1 小程序中请求处理方法是不能传递参数 正确方式 通
  • node js 文件,文件夹,文件流操作

    引入模块 const fs require fs const path require path 读取文件 同步读取 var data fs readFileSync read txt utf 8 console log 同步读取 data
  • LogisticRegression - 参数说明

    LogisticRegression 逻辑回归参数详细说明 参数说明如下 penalty 惩罚项 str类型 可选参数为l1和l2 默认为l2 用于指定惩罚项中使用的规范 newton cg sag和lbfgs求解算法只支持L2规范 L1G
  • SCAU 18724 二叉树的遍历运算

    18724 二叉树的遍历运算 Description 二叉树的三种遍历都可以通过递归实现 如果我们知道一棵二叉树的先序和中序序列 可以用递归的方法求后序遍历序列 输入格式 两行 第一行一个字符串 表示树的先序遍历 第二行一个字符串 表示树的