数据结构代码集训day17(适合考研、自学、期末和专升本)

news2025/1/4 15:51:04

习题来自B站up:白话拆解数据结构


今日习题如下:

1、写出二叉树的前、中、后序遍历

2、写出二叉树的非递归前序和中序遍历


        二叉树有多种存储结构:双亲存储法、孩子兄弟链存储结构,二叉链表存储结构等,一般我们写代码题都用的二叉链表:

typedef struct Bitnode{

    int data;

    struct Bitnode *lchild,*rchild;

}Bitnode,* Bitree;

        如上所示,有三个域,一个数据域,一个左孩子指针域,一个右孩子指针域。对于m叉链表来说,有一个数据域,m个指针域,指针域非空的个数对应该结点的度。下图是一个四叉链表示意图:


题1 

        对于二叉树的前序遍历,我们可以采取递归的做法,因为前序遍历是PLR,所以我们先打印根结点的值,再递归左子树,然后再递归右子树就行了。

    void preorder(Bitree T){    //PLR

        if(T!=NULL){

            printf("%d ",T->data);

           

            preorder(T->lchild);

            preorder(T->rchild);

        }

    }

        对于这种递归的题目,可以采取抽象的方法,下面是一个最简单的二叉树模型:

        对于稍复杂的二叉树,我们可以把他抽象成上述模型,左边的是根结点的左子树,右边是根结点的右子树。

         然后递归左子树preorder(T->lchild)部分,就是把左子树当成一棵新的树,然后把这个新的树再次抽象,一直递归下去!本题的递归出口是T==NULL。


        能理解上述部分,那中序和后序也是一样的,代码如下:

    void inorder(Bitree T){     //LPR

        if(T!= NULL){

            inorder(T->lchild);

           

            printf("%d ",T->data);

            inorder(T->rchild);

        }

    }

    void postorder(Bitree T){       //LRP

        if(T!=NULL){

            postorder(T->lchild);

            postorder(T->rchild);

            printf("%d ",T->data);  

        }

    }

        下面我们来造一个例子,这里采用二叉排序树的递归造法,如下所示:

Bitree createnode(int data){        // 造结点

    Bitree T=(Bitree)malloc(sizeof(Bitnode));

    if (T!=NULL){

        T->data = data;

        T->lchild=NULL;

        T->rchild=NULL;

    }

    return T;

}

Bitree insertnode(Bitree T,int data){        // 二叉排序树插入结点

    if(T==NULL){

        return createnode(data);

    }

    if(data<T->data)

        T->lchild=insertnode(T->lchild,data);

    else if(data>T->data)

        T->rchild=insertnode(T->rchild,data);

    return T;

}

         我们来造这么一棵二叉树:

         我们运行一下:验证是否成功就看中序序列是否有序就行了。

         完整代码如下:

#include <iostream>
#include <cstdio>

#include <ctime>
using namespace std;

typedef struct Bitnode{
    int data;
    struct Bitnode *lchild,*rchild;
}Bitnode,* Bitree;

Bitree createnode(int data){
    Bitree T=(Bitree)malloc(sizeof(Bitnode));
    if (T!=NULL){
        T->data = data;
        T->lchild=NULL;
        T->rchild=NULL;
    }
    return T;
}

Bitree insertnode(Bitree T,int data){
    if(T==NULL){
        return createnode(data);
    }
    if(data<T->data)
        T->lchild=insertnode(T->lchild,data);
    else if(data>T->data)
        T->rchild=insertnode(T->rchild,data);
    return T;
}

// 前中后序遍历
class Solution{
public:
    void preorder(Bitree T){    //PLR
        if(T!=NULL){
            printf("%d ",T->data);
            
            preorder(T->lchild);
            preorder(T->rchild);
        }
    }

    void inorder(Bitree T){     //LPR
        if(T!= NULL){
            inorder(T->lchild);
            
            printf("%d ",T->data);

            inorder(T->rchild);
        }
    }

    void postorder(Bitree T){       //LRP
        if(T!=NULL){
            postorder(T->lchild);
            postorder(T->rchild);

            printf("%d ",T->data);  
        }
    }
};

int main(){
    Bitree T=NULL;
    T=insertnode(T,10);
    T=insertnode(T,7);
    T=insertnode(T,8);
    T=insertnode(T,6);
    T=insertnode(T,12);
    T=insertnode(T,14);
    T=insertnode(T,11);

    Solution s;
    printf("preorder is:");
    s.preorder(T);
    printf("\n");

    printf("inorder is:");
    s.inorder(T);
    printf("\n");

    printf("postorder is:");
    s.postorder(T);
    printf("\n");
    return 0;


}

题2

        非递归一般要借助栈,对于非递归前序遍历二叉树,这个栈实际上是个指针类型的栈,里面存的是指针;借助上面那个图来说明,我们先将T指针(指向根节点的指针)入栈。

         根据前序遍历的特点,根左右,我们入根结点后,立马出栈(需要有一个辅助指针始终指向栈顶),由于temp先指向的T,此时T出栈了,temp依然指向T,根据栈后进先出的特点,我们需要先入temp的右孩子,再入他的左孩子,这样能保持在进入下一轮循环时temp指向左孩子,如下图所示:(10已出栈)。

         下一轮循环开始,我们先出栈栈顶指针,然后重复,先压入temp指针右孩子,再左孩子(7已出栈)。

        此时temp左右孩子都没了,到下一轮直接出栈就行,先出6再出8,最后回退到了12,就开始右子树的遍历了,过程一致。代码如下:

 void preorder_nodigui(Bitree T){

        if(T==NULL)     return;

        stack<Bitree> s;        // 栈存的类型是Bitree指针

        s.push(T);        

        while(!s.empty()){

            Bitnode *temp=s.top();        // 每一轮循环temp指向栈顶

            printf("%d ",temp->data);        

            s.pop();        // 出栈

            if(temp->rchild!=NULL)

                s.push(temp->rchild);

            if(temp->lchild!=NULL)

                s.push(temp->lchild);

        }

    }

        然后是非递归中序遍历,根据LPR的特点,我们需要先将所有的入栈,同样借助,还需要一个辅助指针current, current指针往左下走,将其压栈,直到为空。current指针也需要一直指向栈顶。

        然后我们出栈,先出6,因为6没有孩子,根据LPR原则,直接出P就行!然后current指针就指向7了,把7出栈,因为对于根结点的左子树部分,L已经访问完了,该P部分了,但是7有右孩子,将其右孩子压栈,再出栈(因为8没有孩子)。至此T的左子树部分全部遍历完了,指针也刚好回退到10这里,对于这整棵树,L部分已经完了,该P了,10出栈,然后10有右孩子,将右孩子12入栈,再重复上述过程(入11,出11,出12,入14,出14)。

    void inorder_nodigui(Bitree T){

        if(T==NULL)     return;

        stack<Bitree> s;

        Bitnode *current=T;

        while(current||!s.empty()){

            while(current!=NULL)

            {

                s.push(current);

                current=current->lchild;

            }

            current=s.top();

            printf("%d ",current->data);

            s.pop();

            current = current->rchild;

        }

    }

实践一下:还是可以通过中序序列来判断是否做对。

 

完整代码如下:

#include <iostream>
#include <cstdio>
#include <stack>
#include <ctime>
using namespace std;

typedef struct Bitnode{
    int data;
    struct Bitnode *lchild,*rchild;
}Bitnode,* Bitree;

Bitree createnode(int data){
    Bitree T=(Bitree)malloc(sizeof(Bitnode));
    if (T!=NULL){
        T->data = data;
        T->lchild=NULL;
        T->rchild=NULL;
    }
    return T;
}

Bitree insertnode(Bitree T,int data){
    if(T==NULL){
        return createnode(data);
    }
    if(data<T->data)
        T->lchild=insertnode(T->lchild,data);
    else if(data>T->data)
        T->rchild=insertnode(T->rchild,data);
    return T;
}

class Solution{
public:
    void preorder_nodigui(Bitree T){
        if(T==NULL)     return;
        stack<Bitree> s;
        s.push(T);
        while(!s.empty()){
            Bitnode *temp=s.top();
            printf("%d ",temp->data);
            s.pop();
            if(temp->rchild!=NULL)
                s.push(temp->rchild);
            if(temp->lchild!=NULL)
                s.push(temp->lchild);
        }
    }

    void inorder_nodigui(Bitree T){
        if(T==NULL)     return;
        stack<Bitree> s;
        Bitnode *current=T;
        while(current||!s.empty()){
            while(current!=NULL)
            {
                s.push(current);
                current=current->lchild;
            }
            current=s.top();
            printf("%d ",current->data);
            s.pop();
            current = current->rchild;
        }
    }
};

int main(){
    Bitree T=NULL;
    T=insertnode(T,10);
    T=insertnode(T,7);
    T=insertnode(T,8);
    T=insertnode(T,6);
    T=insertnode(T,12);
    T=insertnode(T,14);
    T=insertnode(T,11);

    Solution s;
    printf("preorder is:");
    s.preorder_nodigui(T);
    printf("\n");

    printf("inorder is:");
    s.inorder_nodigui(T);
    printf("\n");

}

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2109115.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

如何实现一个定时任务?六种策略可实现

目录标题 1、自定义单线程2、JDK ScheduledExecutorService3、 Spring Task4、Quartz5、Elastic-job6、xxl-job最后&#xff1a;思考更上一层1. 高性能2. 高并发3. 高可用 设计方案 1、自定义单线程 上图中&#xff0c;我们启动一个线程&#xff0c;该线程无限循环执行&#xf…

STM32高级定时器生成互补PWM的原理与代码实现

文章目录 前言一 CubeMx配置1.1 TIM1 Mode and Configuration1.2 Paramter Settings 二 程序代码三 仿真分析总结 前言 互补 PWM&#xff08;Complementary PWM&#xff09;是指一对逻辑状态互为反相的 PWM&#xff08;脉冲宽度调制&#xff09;信号。这种信号配置常见于电机控…

SQL进阶技巧:如何利用SQL解决趣味赛马问题?| 非等值关联匹配问题

目录 0 问题描述 1 数据准备 2 问题分析 方法一:先分后合思想 方法2:非等值关联匹配 3 小结 0 问题描述 有一张赛马记录表,如下所示: create table RacingResults ( trace_id char(3) not null,race_date date not null, race_nbr int not null,win_name char(30) n…

探索 Redis Set:命令、编码与应用实践

set 类型 一 . 常见命令1.1 sadd、smembers1.2 sismember1.3 spop、srandmember1.4 smove1.5 srem1.6 集合间操作交集 : sinter、sinterstore并集 : sunion、sunionstore差集 : sdiff、sdiffstore 小结 二 . 内部编码6.3 应用场景6.3.1 使用 Set 来保存用户的标签6.3.2 使用 Se…

android kotlin基础复习 enum

1、kotlin中&#xff0c;关键字enum来定义枚举类型。枚举类型可以包含多个枚举常量&#xff0c;并且每个枚举常量可以有自己的属性和方法。 2、测试代码&#xff1a; enum class Color{RED,YELLOW,BLACK,GOLD,BLUE,GREEN,WHITE }inline fun <reified T : Enum<T>>…

Qt工程实践_06_Qt MSVC2O17编译器下的程序添加VS2017生成的动态链接库方法

文章目录 1. 利用VS2017生成动态链接库1.1 创建C++空项目1.2 添加.h和.cpp内容:添加了一个减法运算1.3 设置动态链接库目标计算机类型1.4 设置项目属性为动态库1.5 生成项目,复制需要的文件2. Qt程序使用VS2017生成的动态链接库方法2.1 创建Widget程序2.2 链接动态库文件2.2.…

蚂蚁SEO|AI养站程序是什么|蚂蚁蜘蛛池

《AI 养站程序&#xff1a;开启网站运营新未来》 在当今数字化时代&#xff0c;网站运营的重要性日益凸显。而 AI 养站程序的出现&#xff0c;为网站运营者带来了全新的机遇与挑战。 一、什么是 AI 养站程序 AI 养站程序是利用人工智能技术&#xff0c;对网站进行自动化管理和优…

MacBook air pro验机流程

由于苹果电脑价格相对较高&#xff0c;用户在网上购置之后&#xff0c;最好对机器要进行一下验机&#xff0c;以确保自己所购置的机器为原厂正品一手机。此外&#xff0c;在网上购置时&#xff0c;注意开相应的发票&#xff0c;方便后续的保修和换机等其他流程。 本文主要是介绍…

中小学生学籍照片(390×480蓝底)手机拍照制作流程说明

近期各地中小学陆续开学&#xff0c;幼升小及小升初一年级新生一般要在十月份之前完成学籍档案采集&#xff0c;其中就包括了新生学籍证件照的采集&#xff08;即学籍照片&#xff09;&#xff0c;部分中部省份使用的学籍照片像素尺寸为390&#xff08;宽&#xff09;480&#…

[C#学习笔记]注释

官方文档&#xff1a;Documentation comments - C# language specification | Microsoft Learn 一、常用标记总结 1.1 将文本设置为代码风格的字体&#xff1a;<c> 1.2 源代码或程序输出:<code> 1.3 异常指示:<exception> 1.4 段落 <para> 1.5 换行&…

Vue+Element多套主题切换

Vue3.x Element Plus与Vue2.x Element ui多套主题的切换方案 demo地址 VueElement更换主题: Vue Element项目&#xff0c;更换几套主题的方案 思路很简单&#xff0c;就是写好每套样式&#xff0c;写个切换功能&#xff0c;切换主题即可 具体实现方案&#xff1a; 准备多…

物联网技术推动灌区智能化管理

物联网技术&#xff0c;作为信息技术革命的重要组成部分&#xff0c;正深刻地改变着传统行业的运作模式&#xff0c;其中在农业灌溉领域的应用尤为显著&#xff0c;为灌区的智能化管理开辟了新径。这一技术通过将传感器、智能网关、大数据分析与云平台紧密融合&#xff0c;实现…

What is Node.JS and its Pros and Cons

What is Node.JS and its Pros and Cons JavaScript is a client-side development tool. Node.js is a server-side development tool. And it’s only a runtime environment based on Chrome V8 so we don’t write some code in Node.js. Pros: JavaScript on a server …

TypeScript 在 Vue.js 中的应用指南

在前端开发中&#xff0c;TypeScript 和 Vue.js 的组合越来越受到青睐。TypeScript 的强类型系统和 Vue.js 的组件化架构相得益彰&#xff0c;可以帮助你编写更可靠和易维护的代码。如果你已经掌握了 TypeScript 的基本语法&#xff0c;但不太确定怎么将它与 Vue.js 配合使用&a…

opencv 实现两个图片的拼接去重功能

基础知识介绍 cv::Mat 是OpenCV库中用来表示图像和矩阵数据的核心类之一。它是一个多维数组&#xff0c;可以存储图像像素数据、矩阵数据以及其他类型的数据。以下是关于 cv::Mat 类的一些详细解释&#xff1a; 构造函数&#xff1a;cv::Mat 类有多个构造函数&#xff0c;可以用…

JavaWeb(后端)

Maven Apache Maven 是一个项目管理和构建工具&#xff0c;它基于项目对象模型(POM)的概念&#xff0c;通过一小段描述信息来管理项目的构建。 Maven的作用 依赖管理&#xff1a;方便快捷的管理项目依赖的资源(jar包)&#xff0c;避免版本冲突问题。 统一项目结构&#xff…

Leetcode22括号生成(java实现)

今天分享的题目是Leetcode22括号生成&#xff0c;具体的题目描述如下&#xff1a; 本道题我们使用的解法是回溯。 解题思路&#xff1a; 我们主要是对括号出现的可能性进行一个收集。 我们以n2举例子&#xff0c;如下图 如果想要合法&#xff0c;那么一定是左括号开始&#…

golang学习笔记05——golang协程池,怎么实现协程池?

推荐学习文档 golang实战大纲golang优秀开发常用开源库汇总golang学习笔记01——基本数据类型golang学习笔记02——gin框架及基本原理golang学习笔记03——gin框架的核心数据结构golang学习笔记04——如何真正写好Golang代码&#xff1f; 协程池是一种用于高效处理任务的机制&…

Claude的小白入门指南

要想快速上手Claude AI&#xff0c;其实并没有那么复杂。作为新一代的AI助手&#xff0c;Claude致力于为用户提供高效、无害、透明的交互体验。这篇入门指南将从Claude AI的特点、主要功能和如何实际操作等几个方面为大家做一个详细的介绍。 Claude AI是什么&#xff1f; Claud…

ssh之登录服务器后,自动进入目录(四十七)

简介&#xff1a; CSDN博客专家、《Android系统多媒体进阶实战》一书作者 新书发布&#xff1a;《Android系统多媒体进阶实战》&#x1f680; 优质专栏&#xff1a; Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 优质专栏&#xff1a; 多媒体系统工程师系列【…