数据背后的力量:揭秘中间件中的二分查找与树结构应用

        平时写业务代码的时候很少写对应的算法,因为很少会在内存中存储大量数据,在需要比较大量数据的查找时,多会依赖的中间件,而中间件底层就应用了很多不同算法,尤其是树结构的查找存储算法,二分查找算法在树里面有大量应用。

二分查找算法

        也称折半查找(Binary Search),它是一种效率较高的查找方法,时间复杂度O(log2n),要求线性表必须采用顺序存储结构,而且表中元素按关键字有序排列 ​​​,在数据量大的情况,比线性查找性能高;在数据量少的情况和线性查找差不多

        线性查找和二分查找效果对比:https://www.cs.usfca.edu/~galles/visualization/Search.html

线性查找

 二分查找

 编码实现

public class BinarySearch {
    public static void main(String[] args) {
        // 定义一个有序数组
        int[] arr = {1, 2, 3, 4, 5, 6, 7, 8, 9};
        // 要查找的元素
        int target = 7;
        // 调用二分查找算法
        int index = binarySearch(arr, target);
        if (index == -1) {
            System.out.println("查找的元素不存在");
        } else {
            System.out.println("查找的元素下标为:" + index);
        }
    }

    /**
     * 二分查找算法
     *
     * @param arr    有序数组
     * @param target 要查找的元素
     * @return 返回元素在数组中的下标,如果没有找到返回-1
     */
    public static int binarySearch(int[] arr, int target) {
        // 记录开始位置
        int start = 0;
        // 记录结束位置
        int end = arr.length - 1;
        // 记录中间位置
        int mid;
        // 循环查找
        while (start <= end) {
            // 计算中间位置
            mid = (start + end) / 2;
            // 比较中间位置的值
            if (arr[mid] == target) {
                // 找到元素,返回下标
                return mid;
            } else if (arr[mid] < target) {
                // 中间元素小于目标元素,则在右边查找
                start = mid + 1;
            } else {
                // 中间元素大于目标元素,则在左边查找
                end = mid - 1;
            }
        }
        // 没有找到元素
        return -1;
    }
}

二叉树

什么是树

        是一种非线性的数据结构,它具有层次结构,根节点是树的最高层,叶节点是最低层。
树可以分为二叉树、平衡树、红黑树、B树、B+树等。

常见术语
 根节点在非空树中没有前驱结点的结点,被称为根节点(注意,只有一个根节点)
父节点每个节点除了根节点外,都有一个父节点
子节点每个节点都可以有1个或多个子节点
叶节点没有子节点的节点称为叶节点
节点的度结点所拥有子树个数,叶子结点就是度为0的结点
树的度树里的各个节点度的最大值
层数根节点为第一层,往下一次递增。
高度(深度)结点的深度指从根节点自顶向下逐层累加至该结点时的深度,树的深度是树中深度最大的结点的深度。
路径从一个节点到另一个节点的序列称为路径
森林由一组树组成的集合称为森林
 

什么是二叉树?

        树有很多种,但是每一个结点最多只有两个子节点的树,被称作为二叉树。二叉树的子节点又分成左节点与右节点。二叉树遍历过程中,每个结点都被访问了一次,时间复杂度是 O(n)
在树的大家族中,二叉树是特别高频使用的树,利用二叉树特性,变种延伸特别多,完全二叉树、满二叉树、二叉查找树(二叉排序树)、平衡二叉树(AVL树)、B-Tree、B+Tree、红黑树(自平衡的二叉查找树,JDK 1.8后的HashMap的存储结构) ...

二叉树的遍历

        二叉树分成根节点、左子树和右子树, 根据根节点什么时候被访问,把二叉树遍历分成三种方式。

        前序遍历:首先访问根节点,然后先是访问左子树,最后才到右子树

        以上图为例,顺序为:A B D H E I J C F G

        中序遍历:首先访问左子树,然后到根节点,最后才到右子树

        以上图为例,顺序为:D H B I E J A F C G

        后序遍历:首先访问左子树,然后到右子树,最后才到根节点

        以上图为例,顺序为:H D I J E B F G C A

二叉树拓展

满二叉树

        每个节点都有0个或者2个子节点的二叉树,叶节点都在最底层的二叉树,节点的总数=2^n-1(n为层数)外观看上去是完整的一个三角形。如节点总数=2^4-1=15

完全二叉树

        只有最下面两层节点的度可以小于2,并且最下层的叶节点集中在靠左连续的边界,只允许最后一层有空缺结点且空缺在右边,完全二叉树需保证最后一个节点之前的节点都齐全;对任一结点,如果其右子树的深度为j,则其左子树的深度必为j或j+1;如果把G节点删除的话,它就不属于完全二叉树了,因为它的叶子结点已经是不连续了。

两者区别

        满二叉树的叶节点都在最底层;完全二叉树只有最下面两层节点的度可以小于2,且最下层的叶节点集中在靠左的边界。

编码分析

树的遍历分为:

1、深度优先遍历:对每一个可能的分支路径深入到不能再深入为止,且每个节点只访问一次,访问完一颗子树再去访问后面的子树;
1)前序遍历:首先访问根节点,然后先是访问左子树,最后才到右子树;
2)中序遍历:首先访问左子树,然后到根节点,最后才到右子树;
3)后序遍历:首先访问左子树,然后到右子树,最后到根节点;

2、广度优先遍历:从根节点到叶子节点的层次关系,一层层横向遍历各个节点,访问树结构的第 n+1 层前必须先访问完第 n 层

编码 

        深度优先遍历 实现二叉树的插入、前序、中序、后序查找。树的定义本身是递归定义,因此采用递归的方法去实现树的三种遍历,容易理解而且代码很简洁

public class BinaryTree {

    // 根节点
    TreeNode root;

    public void insertNode(int data) {
        root = insert(root, data);
    }

    private TreeNode insert(TreeNode treeNode, int data) {
        //如果是空则插入第一个节点
        if (treeNode == null) {
            return new TreeNode(data);
        }
        //插左边
        if (data < treeNode.data) {
            treeNode.leftChild = insert(treeNode.leftChild, data);
        }
        //插右边
        else if (data > treeNode.data) {
            treeNode.rightChild = insert(treeNode.rightChild, data);
        } else {
            treeNode.data = data;
        }
        return treeNode;
    }


    // 前序遍历
    public void preOrder(TreeNode root) {
        if (root == null) {
            return;
        }
        System.out.print(root.data + " ");
        preOrder(root.leftChild);
        preOrder(root.rightChild);
    }

    // 中序遍历
    public void inOrder(TreeNode root) {
        if (root == null) {
            return;
        }
        inOrder(root.leftChild);
        System.out.print(root.data + " ");
        inOrder(root.rightChild);
    }

    // 后序遍历
    public void postOrder(TreeNode root) {
        if (root == null) {
            return;
        }
        postOrder(root.leftChild);
        postOrder(root.rightChild);
        System.out.print(root.data + " ");
    }
}

class TreeNode {
    int data;
    //左节点
    TreeNode leftChild;
    //右节点
    TreeNode rightChild;

    TreeNode(int data) {
        this.data = data;
    }
}

测试

 public static void testBinaryTree(){
        BinaryTree btt=new BinaryTree();
        btt.insertNode(93);
        btt.insertNode(33);
        btt.insertNode(51);
        btt.insertNode(21);
        btt.insertNode(102);
        btt.insertNode(42);
        btt.preOrder(btt.root);
        System.out.println();
        btt.inOrder(btt.root);
        System.out.println();
        btt.postOrder(btt.root);
    }

 输出结果

前序:93 33 21 51 42 102 

中序:21 33 42 51 93 102 

后序:21 42 51 33 102 93 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

状态管理@Prop、@Link装饰器

Prop Link 父子组件进行数据同步化 prop 单向同步 只支持string、number、boolean、enum类型 负组件对象类型&#xff0c;总组件是对象类型 不可以是数组、any 不允许子组件初始化 Link双向同步 父子类型一直&#xff1a;string、number、boolean、enum、object、class以及他们…

centos7 yum 安装mysql8.0.36

一、前置条件&#xff1a;CentOS Mini 安装需要先安装ifconfig 和 wget工具&#xff0c;参考步骤&#xff1a; 1.首先&#xff0c;让我们找出哪个包提供了ifconfig命令。要完成这项任务&#xff0c;输入以下命令&#xff1a; yum provides ifconfig 输出&#xff1a; [rootlo…

Vue3 + Django 前后端分离项目实现密码认证登录

1、功能需求 通常中小型前后端项目&#xff0c;对安全要求不高&#xff0c;也可以采用密码认证方案。如果只用django来实现非常简单。采用 Vue3 前后端分离架构&#xff0c;实现起来稍繁琐一点&#xff0c;好处是可以利用各种前端技术栈&#xff0c;如element-plus UI库来渲染…

Android Jetpack Compose基础之组件的帧渲染

Android Jetpack Compose基础之组件的帧渲染 组合布局LayoutModifier示例 LayoutCompsable示例 绘制CanvasDrawModifierDrawModifier-drawWithContent示例 DrawModifier-drawBehind源码示例 DrawModifier-drawWithCache源码示例 拓展Modifier.graphicsLayer Android View 系统&…

6-191 拓扑排序

一项工程由一组子任务构成,子任务之间有的可以并行执行,有的必须在完成了其他子任务后才能执行。例如,下图表示了一项工程若干子任务之间的先后关系。 编写函数输出所有子任务的拓扑序列。 函数接口定义: Status Push_SeqStack(SeqStack &s, ElemType x)//入栈,x入到…

Polar 2024春季个人挑战赛 Jay17 WP

Polar 2024春季个人挑战赛 Rank&#xff1a;7 【WEB】机器人 开题 起手敏感文件robots.txt 【WEB】PHP反序列化初试 最简单的php反序列化 POC&#xff1a; <?php class Easy{public $name;public function __wakeup(){echo $this->name;} } class Evil{public $evi…

2、Jenkins持续集成-gitlab安装和源码上传

文章目录 1、Gitlab代码托管服务器安装2、源代码上传托管 环境&资源准备 统一采用VMware中安装CentOS7&#xff0c;安装教程&#xff0c;统一设置静态IP资源包都存在于我的资源里面 资源版本&位置 名称机器IP软件代码托管服务器192.168.2.100Gitlab-12.4.2持续集成服…

idm下载器 idm网络加速器 Internet Download Manager(IDM)免激活不弹窗版

idm下载器官方版英文名为Internet Download Manager。是一款界面简单、体积小巧下载速度飞快的下载工具&#xff0c;支持断点续传、多个任务同时下载&#xff0c;同时还支持限速下载&#xff0c;在自义文件类型里还可以自由设置下载的文件类型。 一、IDM软件安装 Internet Dow…

Linux安装Nginx及配置TCP负载均衡

目录 1、安装编译工具及库文件2、下载解压Nginx压缩包3、Ngnix配置Tcp负载均衡4、配置Ngnix的文件5、Nginx启动 1、安装编译工具及库文件 yum -y install make zlib zlib-devel gcc-c libtool openssl openssl-devel pcre-devel2、下载解压Nginx压缩包 wget https://nginx.o…

Docker 安装 Tomcat

目录 1. 总体步骤 2.安装tomcat 2.1 搜索 2.2 拉取 2.3 查看是否拉取到镜像 2.4 运行镜像 2.5 访问tomcat首页 1. 总体步骤 搜索镜像拉取镜像查看镜像启动镜像停止容器移除容器 2.安装tomcat 2.1 搜索 2.2 拉取 docker pull tomcat 2.3 查看是否拉取到镜像 docker …

连接医患的桥梁:深入了解互联网医院APP的开发与优化

当下&#xff0c;互联网技术的不断发展&#xff0c;越来越多的医院和医疗机构开始关注并投入到互联网医院APP的开发与优化中。接下来&#xff0c;小编将与大家共同探讨互联网医院APP的开发与优化。 一、互联网医院APP的开发原则 &#xff08;1&#xff09;用户体验至上 界面设…

卷积篇 | YOLOv8改进之主干网络中引入可变形卷积DConv

前言:Hello大家好,我是小哥谈。可变形卷积模块是一种改进的卷积操作,它可以更好地适应物体的形状和尺寸,提高模型的鲁棒性。可变形卷积模块的实现方式是在标准卷积操作中增加一个偏移量offset,使卷积核能够在训练过程中扩展到更大的范围,从而实现对尺度、长宽比和旋转等各…

git基础-查看提交历史

查看提交历史 在创建了多个提交之后&#xff0c;或者如果克隆了一个具有现有提交历史的存储库&#xff0c;可能会想要回顾一下发生了什么。最基本和强大的工具就是 git log 命令。 运行下git log查看下输出状态 默认情况下&#xff0c;不带任何参数运行 git log 命令会以逆时…

拌合楼管理系统(八) c#海康威视摄像头车牌识别

前言: c#调用海康威视SDK实现车牌识别 原本以为海康威视sdk的Demo里面没有车牌识别的实例,后来发现自己肤浅了,官方是有提供的,只是车牌识别是通过安防布警的方式实现的.程序主动监听,触发告警后获取到车牌信息. 一、接口调用的流程&#xff1a; 首先初始化sdk -> 开…

袁志佳:前端全栈工程师精英班

教程介绍 本套课程涵盖大前端的全部领域&#xff0c;并对传统的Web前端全栈深入教学。如利用前端知识开发 AI、VR、AR、iOS、Android、PC、Server、智能硬件。 同时我们将核心打造 JavaScript语言新发展、Vue源码分析、前端持续集成方案汇总、MV*框架深度分析 、前端图形学、N…

亚稳态及其解决办法

异步电路 亚稳态 亚稳态亚稳态的产生原因什么是同步异步信号怎么消除亚稳态 亚稳态 在数字电路中&#xff0c;每一位数据不是1&#xff08;高电平&#xff09;就是0&#xff08;低电平&#xff09;。当然对于具体的电路来说&#xff0c;并非1&#xff08;高电平&#xff09;就是…

elementary OS7 Ubuntu 22.04中硬盘挂载报错

elementary OS7 Ubuntu 22.04中硬盘挂载报错 背景目标思路解决方法 背景 上周末安装elementaryos7的过程中将windows10的引导文件搞丢了&#xff0c;这两天准备修复一下&#xff0c;保险期间将固态硬盘上的文件备份到移动硬盘上&#xff0c;备份过程中出现报错的问题&#xff…

2024常用Web支付开发讲解教程

课程介绍 本教程为web支付开发&#xff0c;讲解了最常用的两钟支付&#xff1a;支付宝支付和微信支付&#xff0c;服务器配置和API对接&#xff0c;学完本课程可以学会微信支付、和支付宝支付开发。 学习资料 链接&#xff1a;https://pan.baidu.com/s/19WarLP2dO4dFvNbIHLU…

C++类的6个默认成员函数(构造)

C类和对象基础-CSDN博客https://blog.csdn.net/lh11223326/article/details/136834917?spm1001.2014.3001.5501 目录 1.构造函数 概念 特性 2.析构函数 概念 特性 3.拷贝构造函数 概念 特征 4.赋值运算符重载&#xff08;构造实现&#xff09; 运算符重载 赋值运算…

【常见BUG系列】Java 编程中的 NoSuchFieldError 异常:原因与解决方法

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…
最新文章