`
carus
  • 浏览: 30892 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

【zz】二叉树遍历及C语言实现

 
阅读更多

二叉树遍历及C语言实现

已知中序和前序序列,或者已知中序和后序序列,都能够构造一棵二叉树。在本例中,本人用C语言写程序解答了下面两个算法题:

(1)给出一棵二叉树的中序与后序遍历序列,求出它的先序遍历序列。

(2)给出一棵二叉树的中序与先序遍历序列,求出它的后序遍历序列。

知识点扼要回顾:
所谓二叉树的遍历,是指按一定的顺序对二叉树中的每个结点均访问一次,且仅访问一。按照根结点访问位置的不同,通常把二叉树的遍历分为六种:
TLR(根左右), TRL(根右左), LTR(左根右)
RTL(右根左), LRT(左右根), RLT(右左根)

其中,TRL、RTL和RLT三种顺序在左右子树之间均是先右子树后左子树,这与人们先左后右的习惯不同,因此,往往不予采用。余下的三种顺序TLR、LTR和LRT根据根访问的位置不同分别被称为前序遍历、中序遍历和后序遍历。

前序遍历的规律是:输出根结点,输出左子树,输出右子树;  
中序遍历的规律是:输出左子树,输出根结点,输出右子树; 
后序遍历的规律是:输出左子树,输出右子树,输出根结点;

不多说了,看代码吧:)

 

  1. #include <iostream>  
  2. #include <string>  
  3.   
  4. using namespace std;  
  5.   
  6. //存储节点数据,为简便起见,这里选用字符  
  7. typedef char   DATA_TYPE;  
  8.   
  9. typedef struct tagBINARY_TREE_NODE  BINARY_TREE_NODE, *LPBINARY_TREE_NODE;  
  10.   
  11. struct tagBINARY_TREE_NODE  
  12. {  
  13.       DATA_TYPE             data;           //节点数据  
  14.       LPBINARY_TREE_NODE    pLeftChild;     //左孩子指针  
  15.       LPBINARY_TREE_NODE    pRightChild;    //右孩子指针  
  16. };  
  17.   
  18. //  
  19. //函数名称:TreeFromMidPost  
  20. //函数功能:给出一棵二叉树的中序与后序序列,构造这棵二叉树。   
  21. //输入参数:LPBINARY_TREE_NODE & lpNode:二叉树的结点   
  22. //          string mid:存储了二叉树的中序序列的字符串   
  23. //          string post:存储了二叉树的后序序列的字符串   
  24. //          int lm, int rm:二叉树的中序序列在数组mid中的左右边界   
  25. //          int lp, int rp:二叉树的后序序列在数组post中的左右边界  
  26. //  
  27. void TreeFromMidPost(LPBINARY_TREE_NODE & lpNode, string mid, string post, int lm, int rm, int lp, int rp)  
  28. {  
  29.     //构造二叉树结点  
  30.     lpNode = new BINARY_TREE_NODE;  
  31.     lpNode->data = post[rp];  
  32.     lpNode->pLeftChild = NULL;  
  33.     lpNode->pRightChild = NULL;  
  34.   
  35.     int pos = lm;  
  36.   
  37.     while (mid[pos] != post[rp])  
  38.     {  
  39.         pos++;  
  40.     }  
  41.     int iLeftChildLen = pos - lm;  
  42.   
  43.     if (pos > lm)//有左孩子,递归构造左子树  
  44.     {  
  45.         TreeFromMidPost(lpNode->pLeftChild, mid, post, lm, pos - 1, lp, lp + iLeftChildLen - 1);  
  46.     }  
  47.   
  48.     if (pos < rm)//有右孩子,递归构造右子树  
  49.     {  
  50.         TreeFromMidPost(lpNode->pRightChild, mid, post, pos + 1, rm, lp + iLeftChildLen, rp - 1);  
  51.     }  
  52. }  
  53.   
  54. //  
  55. //函数名称:TreeFromMidPre  
  56. //函数功能:给出一棵二叉树的先序与中序序列,构造这棵二叉树。   
  57. //输入参数: BINARY_TREE_NODE & lpNode:二叉树的结点  
  58. //          string mid:存储了二叉树的中序序列的字符串   
  59. //          string pre:存储了二叉树的先序序列的字符串   
  60. //          int lm, int rm:二叉树的中序序列在数组mid中的左右边界   
  61. //          int lp, int rp:二叉树的先序序列在数组pre中的左右边界  
  62. //  
  63. void TreeFromMidPre(LPBINARY_TREE_NODE & lpNode, string mid, string pre, int lm, int rm, int lp, int rp)  
  64. {  
  65.     //构造二叉树结点  
  66.     lpNode = new BINARY_TREE_NODE;  
  67.     lpNode->data = pre[lp];  
  68.     lpNode->pLeftChild = NULL;  
  69.     lpNode->pRightChild = NULL;  
  70.   
  71.     int pos = lm;  
  72.   
  73.     while (mid[pos] != pre[lp])  
  74.     {  
  75.         pos++;  
  76.     }  
  77.     int iLeftChildLen = pos - lm;  
  78.   
  79.     if (pos > lm)//有左孩子,递归构造左子树  
  80.     {  
  81.         TreeFromMidPre(lpNode->pLeftChild, mid, pre, lm, pos - 1, lp + 1, lp + iLeftChildLen);  
  82.     }  
  83.   
  84.     if (pos < rm)//有右孩子,递归构造右子树  
  85.     {  
  86.         TreeFromMidPre(lpNode->pRightChild, mid, pre, pos + 1, rm, lp + iLeftChildLen + 1, rp);  
  87.     }  
  88. }  
  89.   
  90. //先序遍历  
  91. void PreOrder(LPBINARY_TREE_NODE p)  
  92. {  
  93.        if(p != NULL)  
  94.        {  
  95.               cout << p->data;          //输出该结点  
  96.               PreOrder(p->pLeftChild);  //遍历左子树   
  97.               PreOrder(p->pRightChild); //遍历右子树  
  98.        }  
  99. }  
  100.   
  101. //中序遍历  
  102. void MidOrder(LPBINARY_TREE_NODE p)  
  103. {  
  104.        if(p != NULL)  
  105.        {  
  106.               MidOrder(p->pLeftChild);   //遍历左子树   
  107.               cout << p->data;           //输出该结点  
  108.               MidOrder(p->pRightChild);  //遍历右子树  
  109.        }  
  110. }  
  111.   
  112. //后序遍历  
  113. void PostOrder(LPBINARY_TREE_NODE p)  
  114. {  
  115.        if(p != NULL)  
  116.        {  
  117.               PostOrder(p->pLeftChild);  //遍历左子树   
  118.               PostOrder(p->pRightChild); //遍历右子树  
  119.               cout << p->data;           //输出该结点  
  120.        }  
  121. }  
  122.   
  123. //释放二叉树  
  124. void Release(LPBINARY_TREE_NODE lpNode)  
  125. {  
  126.     if(lpNode != NULL)  
  127.     {  
  128.         Release(lpNode->pLeftChild);  
  129.         Release(lpNode->pRightChild);  
  130.         delete lpNode;  
  131.         lpNode = NULL;  
  132.     }  
  133. }  
  134.   
  135.   
  136. int main(int argc, char* argv[])  
  137. {  
  138.     string              pre;            //存储先序序列  
  139.     string              mid;            //存储中序序列  
  140.     string              post;           //存储后序序列  
  141.   
  142.     LPBINARY_TREE_NODE  lpRoot;         //二叉树根节点指针  
  143.   
  144.   
  145.     cout<<"程序1已知二叉树的中序与后序序列,求先序序列"<<endl;  
  146.     cout<<"请先输入中序序列,回车后输入后序序列:"<<endl;  
  147.   
  148.     cin >> mid;  
  149.     cin >> post;  
  150.   
  151.     TreeFromMidPost(lpRoot, mid, post, 0, mid.size()-1, 0, post.size()-1);  
  152.     cout<<"先序遍历结果:";  
  153.     PreOrder(lpRoot);  
  154.     cout<<endl;  
  155.   
  156.     cout<<"中序遍历结果:";  
  157.     MidOrder(lpRoot);  
  158.     cout<<endl;  
  159.   
  160.     cout<<"后序遍历结果:";  
  161.     PostOrder(lpRoot);  
  162.     cout<<endl;  
  163.     Release(lpRoot);  
  164.     cout<<endl;  
  165.   
  166.     cout<<"程序2已知二叉树的中序与先序序列,求后序序列"<<endl;  
  167.     cout<<"请先输入先序序列,回车后输入中序序列:"<<endl;  
  168.     cin >> pre;  
  169.     cin >> mid;  
  170.   
  171.     TreeFromMidPre(lpRoot, mid, pre, 0, mid.size()-1, 0, pre.size()-1);  
  172.     cout<<"先序遍历结果:";  
  173.     PreOrder(lpRoot);  
  174.     cout<<endl;  
  175.   
  176.     cout<<"中序遍历结果:";  
  177.     MidOrder(lpRoot);  
  178.     cout<<endl;  
  179.   
  180.     cout<<"后序遍历结果:";  
  181.     PostOrder(lpRoot);  
  182.     cout<<endl;  
  183.     Release(lpRoot);  
  184.     cout<<endl;  
  185.   
  186.   
  187.     system("pause");  
  188.     return 0;  
  189. }  

 

(1)程序1的输入方式:
已知二叉树的中序与后序序列,求先序序列,请先输入中序序列,回车后输入后序序列:
例如输入:
DGBAECHF
GDBEHFCA
输出:
先序遍历结果:ABDGCEFH
中序遍历结果:DGBAECHF
后序遍历结果:GDBEHFCA

(2)程序2的输入方式:
已知二叉树的先序与中序序列,求后序序列,请先输入先序序列,回车后输入中序序列:
例如输入:
abdefgc
debgfac
输出:
先序遍历结果:abdefgc
中序遍历结果:debgfac
后序遍历结果:edgfbca

最后请看该程序运行效果图:

这是程序1所确定的二叉树图:

这是程序2所确定的二叉树图:

by Loomman, QQ:28077188, MSN: Loomman@hotmail.com QQ裙:30515563 ☆程序天堂☆ 请尊重作者原创,转载注明来自裂帛一剑博客,谢谢合作。

分享到:
评论

相关推荐

    c语言10个数据结构设计实例二叉树建立遍历冒泡排序快速排序等

    c语言10个数据结构设计实例二叉树建立遍历冒泡排序快速排序等提取方式是百度网盘分享地址

    C常用算法程序集非数值计算排序

    在C语言中实现算法,尤其是在非数值计算排序方面,是计算机科学教育和实践中的一项基础工作。 提到“非数值计算排序”,我们通常指的是处理非数值数据的排序算法。这类算法在信息管理、数据整理以及数据库等领域...

    数据结构课程设计 哈夫曼树

    文件"zZ.cpp"很可能是这个实现过程中的源代码,它可能包含了构建哈夫曼树和生成哈夫曼编码的函数。 哈夫曼编码的应用广泛,例如在文本压缩中,通过将出现频率高的字符赋予较短的编码,可以有效提高压缩效率。此外,...

    计算机考研大纲

    - **二叉树的遍历**:掌握前序遍历、中序遍历、后序遍历的方法。 - **线索二叉树**:通过线索化处理二叉树,提高查找效率。 - **二叉排序树**:一种特殊的二叉树,左子树上的所有结点的值均小于它的根结点的值;...

    ABB常用机器人技术参数.pdf

    ABB常用机器人技术参数.pdf

    西门子1200 PLC FB284功能块实现多设备控制:V90伺服、相机角度调整及FANUC机器人DP通讯

    内容概要:本文详细介绍了如何利用西门子1200 PLC及其FB284功能块实现对3台V90伺服电机、相机角度调整以及FANUC机器人的控制。主要内容涵盖FB284功能块的基础参数设置、多台伺服电机的具体控制方法、相机角度调整的实现、DP通讯配置FANUC机器人控制,以及PLC程序注解和触摸屏程序的设计。通过具体代码示例和实际操作步骤,帮助读者理解和掌握这一系列控制技术。 适合人群:具备一定PLC基础知识的工控初学者和技术人员。 使用场景及目标:① 学习并掌握FB284功能块的使用方法;② 实现多台V90伺服电机的协同控制;③ 掌握相机角度调整的技术细节;④ 完成FANUC机器人通过DP通讯的控制配置;⑤ 提高PLC程序的可读性和易维护性。 其他说明:文中提供了丰富的代码片段和配置示例,便于读者实践操作。此外,还分享了一些实际项目中的经验和技巧,有助于提高项目的稳定性和效率。

    《计算机常用工具软件(第3版)》第6章--图形图像工具.ppt

    《计算机常用工具软件(第3版)》第6章--图形图像工具.ppt

    未来产业全球未来产业新赛道布局与发展策略分析:涵盖人工智能、量子科技、氢能等关键技术领域

    内容概要:本文由《未来产业新赛道研究报告》整理而成,涵盖了未来产业在全球范围内的发展态势和竞争形势。报告指出,引领型国家通过全方位体制机制创新,在先进制造、人工智能、量子科技、新一代通信等领域建立了全面领先优势。文中引用了麦肯锡和GVR的数据,预测了人工智能和人形机器人等未来产业的巨大经济潜力。报告还详细介绍了国外和国内对未来产业赛道的重点布局,如量子科技、人工智能、先进网络和通信技术、氢能与储能、生物技术等。此外,报告列举了中国重点省市如北京、上海等的具体发展方向,以及知名研究机构对未来产业热点的分析。最后,报告提出了构建我国未来产业重点赛道目录的建议,包括通用人工智能、高级别自动驾驶、商业航天、人形机器人、新型储能、低空经济、清洁氢、算力芯片、细胞与基因治疗和元宇宙等十大重点赛道。 适用人群:对科技趋势和未来产业发展感兴趣的政策制定者、投资者、企业家和研究人员。 使用场景及目标:①帮助政策制定者了解全球未来产业发展动态,为政策制定提供参考;②为企业提供未来产业布局的方向和重点领域;③为投资者提供投资决策依据,识别未来的投资机会;④为研究人员提供未来科技发展趋势的全景图。 其他说明:报告强调了未来产业在全球经济中的重要性,指出了中国在未来产业布局中的战略定位和发展路径。同时,报告呼吁加强国家顶层设计和行业系统谋划,探索建立未来产业技术预见机制,深化央地联动,推动未来产业高质量发展。

    《网络设备安装与调试(神码版)》2交换机的配置.pptx

    《网络设备安装与调试(神码版)》2交换机的配置.pptx

    自动驾驶路径规划:Lattice算法中的参考线、Frenet坐标系及多项式拟合的Matlab与C++实现

    内容概要:本文详细介绍了自动驾驶路径规划中Lattice算法的基础部分,主要包括三个关键概念和技术实现:参考线生成、Frenet坐标系转换和五次多项式拟合。首先解释了参考线的作用及其生成方法,如三次样条插值和平滑曲线生成。其次探讨了Frenet坐标系的优势,展示了如何将笛卡尔坐标系下的车辆位置投影到参考线上,从而简化路径规划问题。最后讨论了五次多项式的应用,强调其能够确保轨迹的光滑性和舒适性,并提供了详细的Matlab和C++代码实现。 适合人群:对自动驾驶技术感兴趣的开发者、研究人员以及有一定编程基础并希望深入了解路径规划算法的人群。 使用场景及目标:适用于研究和开发自动驾驶系统,特别是进行路径规划模块的设计与实现。主要目标是帮助读者掌握Lattice规划的基本原理和技术细节,以便应用于实际工程项目中。 其他说明:文中不仅有理论讲解,还附带了大量的代码实例,便于读者理解和实践。此外,作者提醒了一些常见的陷阱和注意事项,如避免过拟合、选择合适的插值算法等。

    《网络操作系统(Linux)》项目4-磁盘管理.pptx

    《网络操作系统(Linux)》项目4-磁盘管理.pptx

    《计算机应用基础实训指导》实训十八-PowerPoint-2010的动画和切换.pptx

    《计算机应用基础实训指导》实训十八-PowerPoint-2010的动画和切换.pptx

    安川机器人DX100使用说明书.1.pdf

    安川机器人DX100使用说明书.1.pdf

    《计算机专业英语》Unit-3-What-is-Hardware.ppt

    《计算机专业英语》Unit-3-What-is-Hardware.ppt

    汇川H5U-A16自动贴布网胶机的PLC与威纶通触摸屏集成及优化

    内容概要:本文详细介绍了汇川H5U-A16自动贴布网胶机的PLC控制系统及其与威纶通触摸屏的集成方法。主要内容涵盖伺服轴控制、气缸动作、矩阵托盘管理、OEE统计等方面的编程技巧和优化措施。文中展示了如何将复杂的硬件动作抽象为可复用的功能块(FB),并通过参数配置实现灵活的系统控制。此外,还讨论了如何利用威纶通触摸屏进行实时监控和数据分析,以及如何通过合理的IO表管理和注释提高系统的可维护性和扩展性。 适合人群:从事工业自动化领域的工程师和技术人员,尤其是熟悉PLC编程和触摸屏应用的专业人士。 使用场景及目标:适用于需要开发或优化自动贴布网胶机及其他类似自动化设备的企业。主要目标是提升设备的可靠性和效率,降低维护成本,缩短开发周期。 其他说明:本文不仅提供了具体的编程示例,还分享了许多实战经验和技巧,如如何避免常见的错误和陷阱,如何应对特定硬件特性的挑战等。这些内容对于理解和掌握工业自动化系统的开发非常有价值。

    电力系统暂态稳定性分析:基于Matlab/Simulink的故障仿真与优化

    内容概要:本文详细介绍了利用Matlab和Simulink进行电力系统暂态稳定性分析的方法和技术。首先构建了一个单机无穷大系统的仿真模型,涵盖了同步电机、无穷大电网、输电线路等基础模块的搭建。接着深入探讨了不同类型故障(如短路、断线)的配置方法及其对系统稳定性的影响。针对常见的暂态问题,提出了多种解决方案,包括并联补偿器的应用、自动重合闸的设计以及仿真加速技巧。同时,通过具体案例展示了如何调整关键参数来优化系统性能,确保暂态过程中系统的稳定性和可靠性。 适合人群:从事电力系统研究与开发的技术人员,尤其是对电力系统暂态稳定性感兴趣的工程师和研究人员。 使用场景及目标:适用于需要评估电力系统在突发故障情况下的稳定性的场合,帮助用户掌握故障仿真技术,优化系统设计,提高电力系统的可靠性和安全性。 其他说明:文中提供的代码片段和仿真技巧均经过实际验证,能够显著提升仿真的效率和准确性。建议读者结合自己的项目需求灵活应用相关技术和方法。

    FPGA电机控制:基于Verilog与Nios2的永磁同步电机SVPWM控制系统设计

    内容概要:本文详细介绍了利用FPGA实现永磁同步电机(SPM)的SVPWM控制系统的具体实现方法。系统采用Verilog进行底层硬件时序控制,包括SVPWM模块中的扇区判断、PWM生成以及死区时间控制等;Nios2软核处理器则用于执行控制算法,如磁场定向控制(FOC)、Clarke变换和PID调节器。两者通过Avalon总线连接,实现高效的软硬件协同工作。此外,文中还讨论了一些常见的调试技巧和优化方法,如定点数运算、硬件CRC校验模块的应用等。 适合人群:具备一定FPGA开发经验和电机控制理论基础的技术人员,尤其是从事嵌入式系统开发、自动化控制领域的工程师。 使用场景及目标:适用于需要高精度、高性能电机控制的应用场合,如工业自动化设备、机器人关节控制等。目标是通过软硬件协同设计提高系统的实时性和可靠性,降低电流谐波失真,增强抗干扰能力。 其他说明:文中提供了完整的工程源码和技术细节,有助于读者深入理解和实践。同时,作者分享了许多实用的经验教训,帮助读者避开常见陷阱,提高开发效率。

    《移动商务网页设计与制作》第11章--Web-Worker-处理线程.ppt

    《移动商务网页设计与制作》第11章--Web-Worker-处理线程.ppt

    chromedriver-win64-135.0.7049.114.zip

    chromedriver-win64-135.0.7049.114.zip

    《计算机系统维护》第14章--硬盘分区的调整.ppt

    《计算机系统维护》第14章--硬盘分区的调整.ppt

Global site tag (gtag.js) - Google Analytics