java数据结构与算法 --- 第十章 数结构基础

news2025/7/15 1:24:20

第十章 树结构基础

I 引和基本概念

为什么需要树结构?
数组,查询快,增删慢
链表…
而树结构,同时提高查询和增删!

基本概念
术语: 有手就行

II 二叉树

1.概念:

二叉树:每个节点最多有两个子节点的数叫二叉树
满二叉树: 所有叶子节点都在最后一,结点的总数是2^n-1(n是层数)
完全二叉树:这个概念…怎么说呢,叶子几点只在最后或者倒数第二层,并且最后一层都是左边排列

完全二叉树看起来不完全,为什么叫做完全二叉树呢?
因为,涉及到顺序存储,空间充分利用

2.前中后序遍历,查找

前中后序,针对的是父节点的遍历时机(中序为例:左子节点遍历,父节点,右子结点遍历)

思路:
i 节点类里面完成前中后序遍历的具体操作操作
ii 创建二叉树类,完成操作管理,创建二叉树确定具体的根节点
iii 测试类,手动创建二叉树

上代码:

public class BinaryTreeDemo {
    @Test
    public void test() {
        Node root1 = new Node(1, "1");
        Node node2 = new Node(2, "2");
        Node node3 = new Node(3, "3");
        Node node4 = new Node(4, "4");
        Node node5 = new Node(5, "5");

        root1.setLeft(node2);
        root1.setRight(node3);
        node3.setLeft(node5);
        node3.setRight(node4);

        BinaryTree binaryTree = new BinaryTree();
        binaryTree.setRoot(root1);

        binaryTree.preOrder();//1,2,3,5,4
        System.out.println();
//        binaryTree.infixOrder();//2,1,5,3,4
//        System.out.println();
//        binaryTree.postOrder();//2,5,4,3,1
//        binaryTree.preOrderSearch(2);
    }
}

//定义二叉树
class BinaryTree {
    private Node root;

    public Node getRoot() {
        return root;
    }

    public void setRoot(Node root) {
        this.root = root;
    }

    //前序
    public void preOrder() {
        if (this.root != null) {
            this.root.preOrder();
        } else {
            System.out.println("二叉树为空~~");
        }
    }

    //中
    public void infixOrder() {
        if (this.root != null) {
            this.root.infixOrder();
        } else {
            System.out.println("二叉树为空~~");
        }
    }

    //后
    public void postOrder() {
        if (this.root != null) {
            this.root.postOrder();
        } else {
            System.out.println("二叉树为空~~");
        }
    }

    //前序查找
    public void preOrderSearch(int no) {
        Node resnode = null;
        if (this.root != null) {
            resnode = this.root.preOrderSearch(no);
        }
        if (resnode != null) {
            System.out.println("查找结果-->" + resnode);
        } else {
            System.out.println("find nothings");
        }
    }
}

//创建结点
class Node {
    private int no;
    private String name;
    private Node left; //默认是null
    private Node right; //默认是null

    public Node(int no, String name) {
        this.no = no;
        this.name = name;
    }

    public int getNo() {
        return no;
    }

    public void setNo(int no) {
        this.no = no;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Node getLeft() {
        return left;
    }

    public void setLeft(Node left) {
        this.left = left;
    }

    public Node getRight() {
        return right;
    }

    public void setRight(Node right) {
        this.right = right;
    }

    @Override
    public String toString() {
        return "Node{" +
                "no=" + no +
                ", name='" + name + '\'' +
                '}';
    }

    //编写前序遍历的方法
    public void preOrder() {
        System.out.println(this);
        if (this.left != null) {
            this.left.preOrder();
        }
        if (this.right != null) {
            this.right.preOrder();
        }
    }

    //中序遍历
    public void infixOrder() {
        if (this.left != null) {
            this.left.infixOrder();
        }
        System.out.println(this);
        if (this.right != null) {
            this.right.infixOrder();
        }
    }

    //后序遍历
    public void postOrder() {
        if (this.left != null) {
            this.left.postOrder();
        }
        if (this.right != null) {
            this.right.postOrder();
        }
        System.out.println(this);
    }

    //前序遍历查找
    public Node preOrderSearch(int no) {
        if (this.no == no) {
            return this;
        }
        Node resNode = null;
        if (this.left != null) {
            resNode = this.left.preOrderSearch(no);
        }
        if (resNode != null) {
            return resNode;
        }
        if (this.right != null) {
            resNode = this.right.preOrderSearch(no);
        }
        return resNode;
    }
}

3.删除结点

叶子节点,删除结点;
非叶子结点,删除子树

思路:
因为二叉树是单向的,所以判断当前节点的子节点是否是需要删除的结点(而不是判断当前节点是不是需要删除的结点
i 针对左子节点(非空且是待删除),this.left = null;右子节点同理;
ii 如果没有删除结点,则左/右子树递归
iii 在节点类里面加入删除方法,没考虑根节点,要在二叉树类里面单独写一段判断root是否是待删除的条件

上代码:
节点类中的删除方法:

    //删除结点
    public void delNode(int no) {
        if (this.left != null && this.left.no == no) {//左子结点是待删除结点
            this.left = null;
            return;
        }
        if (this.right != null && this.right.no == no) {//右子节点 是待删除结点
            this.right = null;
            return;
        }

        //如果左右子结点都不是待删除结点,遍历左右子树
        if (this.left != null) {
            this.left.delNode(no);
        }
        if (this.right != null) {
            this.right.delNode(no);
        }
    }

二叉树类中的方法:

    //删除结点
    public void delNode(int no) {
        if (root != null && root.getNo() == no) {//判断root结点是否是待删除结点(节点类的删除没把root包含进去)
            root = null;
        }
        root.delNode(no);
    }

4.顺序存储二叉树

概念: 数组可以转化成树, 数可以转换成数组
在这里插入图片描述

顺序存储二叉树特点: (n是数组下标 对应二叉树的下标
i 通常只考虑完全二叉树
ii 第n个元素的左子节点:2*n+1 ;右子节点:2*n+2
iii 第n个元素是父节点: (n-1)/2

上代码:

public class ArrBinaryTreeDemo {
    @Test
    public void test() {
        int[] arr = {1, 2, 3, 4, 5, 6, 7};
        ArrBinaryTree arrBinaryTree = new ArrBinaryTree(arr);

        arrBinaryTree.preOrder();//1,2,4,5,3,6,7

    }
}

class ArrBinaryTree {
    private int[] arr;

    public ArrBinaryTree(int[] arr) {
        this.arr = arr;
    }

    //方法重载,(针对第一步下标是0的,
    public void preOrder(){
        preOrder(0);
    }

    /**
     * 顺序存储-前序遍历
     *
     * @param index 数组下标
     */
    public void preOrder(int index) {
        if (arr == null || arr.length == 0) {
            System.out.println("数组为空,不能进行二叉树的前序遍历");
            return;
        }
        //输出当前的元素
        System.out.println(arr[index]);

        //左递归
        if ((index * 2 + 1) < arr.length) {
            preOrder((index * 2 + 1));
        }
        //右递归
        if ((index * 2 + 2) < arr.length) {
            preOrder((index * 2 + 2));
        }
    }
}

个人小结:
顺序存储二叉树,并没有实际创建结点. 而是先接受数组,按照二叉树的遍历方式输出.

5.线索化二叉树

引:二叉树的叶子结点,存在指针的浪费

基本介绍:
n个结点的二叉树有 n+1 个空指针域
利用二叉树的空指针域指向该结点的在某次遍历次序下的前驱或者后继结点的指针,叫做线索化二叉树(分,前中后序线索化二叉树)
如图:
在这里插入图片描述

实现思路:
线索化二叉树后,node结点的left和right存在两种情况:
i left可能指向左子树,也可能指向前驱
ii rigth…
传入一个node结点,定义一个pre结点(指向的是node的前驱)
如果传入的node的left是空,则让他指向前驱;同时每次线索化一个节点后,让pre指向node(记录下一个传入node参数的前驱结点),以此通过pre操作后继结点的指向!

上核心代码:
ndoe结点类:(在原来的基础上,加上了leftTyperightType,用来一个结点的left/right是指向左子树还是前驱/…
binaryTree创建二叉树类:(在原来的基础上,加入前驱结点pre,加入线索化的方法

//定义二叉树
class BinaryTree {
    
    //中序遍历
	....

	//前驱节点 !!! 注意每次线索化结点后更新...
    private Node pre = null;

    //线索化二叉树--中序
    public void threadedNodes(Node node) {
        if (node == null) {//不能线索化
            return;
        }

        //(一)线索化左子树
        threadedNodes(node.getLeft());

        //(二)线索化当前结点 -- 中序结果:(8,3,10,1,14,6)
        //处理当前节点的前驱节点 (以8 为例 8.left->null, 8.leftType = 1(前驱)
        if (node.getLeft() == null) {
            //让当前结点的左指针指向前驱节点
            node.setLeft(pre);
            //修改当前节点的左指针类型为1(指向的前驱节点)
            node.setLeftType(1);
        }

        //处理后继节点
        if (pre != null && pre.getRight() == null) {
            pre.setRight(node);
            pre.setRightType(1);
        }

        //处理每个结点后,让当前结点是下一个结点的前驱节点
        pre = node;

        //(三)线索化右子树
        threadedNodes(node.getRight());

    }
    //线索化重载
    public void threadedNodes(){
        threadedNodes(root);
    }
}

//结点类  :在基础上增加一个判断线索化相关的变量
class Node {

	....

    //中序线索化二叉树 说明:
    // --如果leftType/rightType为 0,表示指向左/右子树;为 1,则表示指向 前驱/后继结点
    private int leftType;
    private int rightType;

    //中序遍历
	....
	
	....
}

编程小结:
1.线索化二叉树,判断是否是叶子结点,很关键
2.每次线索化一个结点后,一定要把他设置为pre;因为需要通过pre设置后继节点
另外,关于线索化画出连线:只看叶子结点,根据中序遍历结果查看10,14等,可以发现和1的画线关系(只看叶子结点!!) 如下图:
在这里插入图片描述

6.遍历线索化二叉树

1.分析:
因为线索化之后,各个节点的指向有变化的,原来的遍历方法(判断左右子节点是否是空)不能使用了. 需要新的遍历方式遍历线索化二叉树,各个节点通过线性的方式遍历,提高了遍历的效率.

2.上代码:
思路: – 循环
i 从根结开始找到第一个 leftType == 1
ii 打印
iii 当前结点的右指针指向的是 后继节点(rightType==1 ),就一直输出
iiii 更新这次遍历的结点 (tempNode = tempNode.getRight();)

//线索化遍历二叉树 -- 中序(8,3,10,1,14,6)
public void threadedList() {
    Node tempNode = root;//从root开始遍历
    while (tempNode != null) {
        //循环,找到第一个为leftType==1的 -->8
        while (tempNode.getLeftType() == 0) {
            tempNode = tempNode.getLeft();
        }

        //打印
        System.out.println(tempNode);

        while (tempNode.getRightType() == 1) {
        //如果当前结点的右指针指向的是 后继节点,就一直输出
            tempNode = tempNode.getRight();
            System.out.println(tempNode);
        }

        //替换这个遍历的结点
        tempNode = tempNode.getRight();
    }
}

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

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

相关文章

jeecg-boot中上传图片到华为云obs云存储中

大家好&#xff0c;我是雄雄&#xff0c;欢迎关注微信公众号&#xff1a;雄雄的小课堂。 前言 jeecg-boot框架中&#xff0c;其实对接的功能还是挺多的&#xff0c;其中就有文件云存储服务器&#xff0c;不过是阿里云oss的&#xff0c;那如果我们使用的是七牛云&#xff0c;或…

通过TortoiseGit钩子实现提交前检查作者信息是否正确

1、需求背景 从事嵌入式开发的人运行软件依赖于特定的电脑硬件&#xff0c;可能会存在多人在同一台电脑上开发的需求。 我们使用git进行软件代码版本管理&#xff0c;通过提交时的用户名和邮箱区分某次代码是哪个人提交的信息。git自身支持提交的时临时一次设置成其他的用户信…

01.OpenWrt-写在前面

01.OpenWrt-写在前面 1.1 如何学好OpenWrt OpenWrt究竟应该怎么学,这是我一直在思考的问题! 谈到OpenWrt有相关软硬件知识的人会想到路由器,路由器是OpenWrt系统最主要的使用场景.OpenWrt是基于Linux系统构建起来的,所以其他Linux系统能够做的事情OpenWrt都是可以做到. Ope…

GD32F4(10):GD32转RS422在115200下接收乱码分析

GD32F450&#xff1a;串口转RS485在115200下接收乱码 文章目录GD32F450&#xff1a;串口转RS485在115200下接收乱码1. 知识储备2. 环境3. 操作4. 插入一个知识点&#xff1a;不同MCU串口ip核实现原理4.1 首先我们来看一下STM32f的串口是怎样识别数据的4.2 GD32F4串口识别5. 我的…

毕业设计-基于机器视觉道路视频车道线检测

目录 前言 课题背景和意义 实现技术思路 摄像机校准 ​编辑 透视变换 车道像素查找 识别车道面积 实现效果图样例 前言 &#x1f4c5;大四是整个大学期间最忙碌的时光,一边要忙着备考或实习为毕业后面临的就业升学做准备,一边要为毕业设计耗费大量精力。近几年各个学校要…

电商商家必用的4大TikTok营销变现增长技巧

数据显示&#xff0c;TikTok是目前最热门的海外社交媒体平台&#xff0c;已超过多家老牌社媒平台。由于是个短视频平台&#xff0c;在当下环境里具有很强的营销推广优势&#xff0c;很多电商商家都会选择在TikTok营销产品。那电商商家必用哪些TikTok营销变现增长技巧呢&#xf…

基于决策树的智能网络安全入侵检测模型

基于决策树的智能网络安全入侵检测模型学习目标&#xff1a;学习内容&#xff1a;该论文模型下载数据集参考论文&#xff1a;综述/调查&#xff1a;申明&#xff1a; 未经许可&#xff0c;禁止以任何形式转载&#xff0c;若要引用&#xff0c;请标注链接地址。 全文共计4077字&…

设备全生命周期管理第一股凌雄科技上市,京东、腾讯等长期看好

“设备全生命周期管理第一股”凌雄科技&#xff08;小熊U租&#xff09;于昨日正式在港交所挂牌上市。本次上市&#xff0c;凌雄科技的IPO发行价为7.60港元/股&#xff0c;募集资金净额约为3.372亿港元&#xff0c;上市时的总市值达到26.85亿港元。 特别说明的是&#xff0c;凌…

适用更多会议场景,华为云会议的分组讨论功能来了!

适用更多会议场景&#xff0c;华为云会议的分组讨论功能来了&#xff01; 如今&#xff0c;线上沟通成为常态&#xff0c;线上会议更是成为工作推进过程中不可缺少的环节。但在一些场景中&#xff0c;例如在跨部门协调&#xff0c;沙龙研讨&#xff0c;教育培训或者招聘面试时&…

索引生命周期管理ILM看完不懂你锤我

阅读完本文你可以学到什么是索引生命周期管理&#xff0c;各个阶段可以做的操作以及如何使用索引模版使用索引生命周期策略&#xff0c;下面就跟我一起来吧 基础理论篇 索引生命周期管理&#xff08;ILM&#xff09;是一种可以让我们随着时间推移自动化的管理索引的一种方式。…

SpringBoot SpringBoot 原理篇 1 自动配置 1.1 bean 的加载方式【一】

SpringBoot 【黑马程序员SpringBoot2全套视频教程&#xff0c;springboot零基础到项目实战&#xff08;spring boot2完整版&#xff09;】 SpringBoot 原理篇 文章目录SpringBootSpringBoot 原理篇1 自动配置1.1 bean 的加载方式【一】1.1.1 环境准备1.1.2 第一种方式1.1.3 第…

Go-Excelize API源码阅读(三十四)——RemoveRow

Go-Excelize API源码阅读&#xff08;三十四&#xff09;——RemoveRow 开源摘星计划&#xff08;WeOpen Star&#xff09; 是由腾源会 2022 年推出的全新项目&#xff0c;旨在为开源人提供成长激励&#xff0c;为开源项目提供成长支持&#xff0c;助力开发者更好地了解开源&a…

OpenGL原理与实践——核心模式(五):颜色、基础光照、Phong模型、材质与光

目录 颜色相关理论 什么是颜色 如何计算颜色&#xff1f; 简单实现 Phong光照模型——局部光照模型 环境光 ​编辑 漫反射 镜面反射 材质与光 材质与纹理的关系 材质在shader的体现 材质属性与光属性 光在shader的体现 整体源码实现及渲染结果 关键代码 shade…

软件测试员如何在恶劣的内卷环境下脱颖而出?

内卷&#xff0c;是现在热度非常高的一个词汇&#xff0c;随着热度不断攀升&#xff0c;隐隐到了“万物皆可卷”的程度。 我一个很要好的朋友&#xff0c;现在就读大三&#xff0c;像很多大学生一样面临着能否顺利毕业的压力和考证的焦虑&#xff0c;看着寝室四个人每天都在玩&…

Vue--》混合文件使用以及ref的引用讲解

目录 mixin混合 前言 ref的引用DOM ref引用组件 mixin混合 在日常开发中&#xff0c;当我们开发的各种组件可能会有相同的内容&#xff0c;我们可以将相同的内容在各个相对应的组件内删除&#xff0c;然后放在同一个配置里。所谓混合&#xff1a;两个或多个组件共享一个配…

数据挖掘,在商业智能BI领域的运用

数据挖掘在商业领域&#xff0c;特别是在零售业的运用是比较成功的。由于各业务系统的普遍使用&#xff0c;再加上商业智能BI的可视化分析&#xff0c;企业可以收集到大量关于购买情况的数据&#xff0c;并且数据量在不断激增。利用数据挖掘技术可以为经营管理人员提供正确的决…

Java 集合

目录 一、概念 二、接口 2.1、 集合接口 2.2、 Set 接口 2.2.1 zise方法 2.2.2 isEmpty 方法 2.2.3 contains 方法 2.2.4 Iterator 方法 2.2.5 toArray 方法 2.2.6 add 方法 2.2.7 remove 方法 2.2.8 containsAll 方法 2.2.9 containsAll 方法 2.2.10 ret…

数据仓库基础

文章目录1 数据仓库1.1 数据仓库为何而来1.2 数据仓库主要特征1.2.1 面向主题1.2.2 集成性1.2.3 非易失性1.2.4 时变性1.3 数据仓库、数据库、数据集市1.3.1 OLTP1.3.2 OLAP1.3.3 OLTP和OLAP的对比1.3.4 数据库和数据仓库的区别1.3.5 数据仓库和数据集市的区别1.4 数据仓库分层…

Spring Boot Admin2 自定义异常监控

其他相关文章&#xff1a; Spring Boot Admin 参考指南SpringBoot Admin服务离线、不显示健康信息的问题Spring Boot Admin2 EnableAdminServer的加载Spring Boot Admin2 AdminServerAutoConfiguration详解Spring Boot Admin2 实例状态监控详解Spring Boot Admin2 自定义JVM监控…

Java项目:JSP旅游产品销售管理系统

作者主页&#xff1a;源码空间站2022 简介&#xff1a;Java领域优质创作者、Java项目、学习资料、技术互助 文末获取源码 项目介绍 本项目分为前后台&#xff0c;分为管理员与普通用户两种角色&#xff0c;管理员登录后台&#xff0c;普通用户登录前台&#xff1b; 管理员角色…