【算法系列 | 13】深入解析查找算法之—树表查找

引言

查找算法在计算机科学中扮演着至关重要的角色。它们的效率直接影响到系统的性能和用户体验。树表查找(Tree-based Search)是一类基于树结构的查找算法,广泛应用于各类数据结构和数据库系统中。
本文将深入介绍树表查找算法的原理优缺点复杂度分析使用场景,并提供JavaPython的实现示例。

一、树表查找算法的原理

树表查找算法主要基于二叉查找树(Binary Search Tree, BST)、平衡二叉树(如AVL树和红黑树)、B树及其变种。它们的共同点是使用树结构来组织数据,使得查找、插入和删除操作的时间复杂度能够保持在较低的水平。

1.1 二叉查找树(BST)

二叉查找树是一种每个节点最多有两个子节点的树结构。对于每个节点,其左子树所有节点的值都小于该节点的值,右子树所有节点的值都大于该节点的值。这样的结构使得查找操作能够通过逐层比较,高效地缩小查找范围。

1.2 平衡二叉树

平衡二叉树通过各种机制(如旋转操作)保持树的平衡,确保树的高度不会超过O(log n)。常见的平衡二叉树包括AVL树和红黑树。AVL树通过维护每个节点的平衡因子来实现平衡,红黑树通过颜色标记和旋转操作来维持近似平衡。

1.3 B树及其变种

B树是一种广泛应用于数据库和文件系统的多路平衡查找树。B树的每个节点可以有多个子节点和关键字,能够在磁盘IO操作中更高效地进行查找。B+树和B*树是B树的常见变种,具有更高的空间利用率和查询效率。

二、优缺点

2.1 优点

  1. 高效的查找性能:平衡二叉树和B树的查找、插入、删除操作的时间复杂度为O(log n)。
  2. 动态性:支持动态插入和删除操作,适用于需要频繁更新的数据集。
  3. 空间利用率高:特别是B树及其变种在磁盘IO操作中具有高效的空间利用率。

2.2 缺点

  1. 实现复杂性:平衡二叉树和B树的实现相对复杂,需要维护平衡性或节点分裂与合并。
  2. 空间开销:在某些情况下,树节点需要存储额外的信息(如父节点指针、平衡因子、颜色标记等),增加了空间开销。

三、复杂度分析

  1. 查找:O(log n)
  2. 插入:O(log n)
  3. 删除:O(log n)

这些复杂度主要源于树的高度在平衡情况下为O(log n),使得操作只需访问较少的节点。

3.1 使用场景

  1. 数据库索引:B树和B+树广泛用于数据库的索引结构。
  2. 内存中数据结构:如Java的TreeMap和TreeSet使用红黑树作为底层数据结构。
  3. 文件系统:许多文件系统使用B树变种来管理文件和目录。

四、代码示例实现

4.1 Java 实现示例:红黑树

下面的Java代码展示了一个简化的红黑树实现,包括插入和查找操作。

import java.util.*;public class RedBlackTree<K extends Comparable<K>, V> {private static final boolean RED = true;private static final boolean BLACK = false;private class Node {K key;V value;Node left, right;boolean color;Node(K key, V value, boolean color) {this.key = key;this.value = value;this.color = color;}}private Node root;public V get(K key) {Node x = root;while (x != null) {int cmp = key.compareTo(x.key);if (cmp < 0) x = x.left;else if (cmp > 0) x = x.right;else return x.value;}return null;}public void put(K key, V value) {root = put(root, key, value);root.color = BLACK;}private Node put(Node h, K key, V value) {if (h == null) return new Node(key, value, RED);int cmp = key.compareTo(h.key);if (cmp < 0) h.left = put(h.left, key, value);else if (cmp > 0) h.right = put(h.right, key, value);else h.value = value;if (isRed(h.right) && !isRed(h.left)) h = rotateLeft(h);if (isRed(h.left) && isRed(h.left.left)) h = rotateRight(h);if (isRed(h.left) && isRed(h.right)) flipColors(h);return h;}private boolean isRed(Node x) {if (x == null) return false;return x.color == RED;}private Node rotateLeft(Node h) {Node x = h.right;h.right = x.left;x.left = h;x.color = h.color;h.color = RED;return x;}private Node rotateRight(Node h) {Node x = h.left;h.left = x.right;x.right = h;x.color = h.color;h.color = RED;return x;}private void flipColors(Node h) {h.color = RED;h.left.color = BLACK;h.right.color = BLACK;}public static void main(String[] args) {RedBlackTree<Integer, String> tree = new RedBlackTree<>();tree.put(1, "one");tree.put(2, "two");tree.put(3, "three");System.out.println(tree.get(1));  // 输出: oneSystem.out.println(tree.get(2));  // 输出: twoSystem.out.println(tree.get(3));  // 输出: three}
}
代码讲解
  1. Node类:定义了红黑树的节点结构,包括键、值、左右子节点和颜色。
  2. isRed方法:判断节点是否为红色。
  3. rotateLeft方法:左旋操作,用于保持树的平衡。
  4. rotateRight方法:右旋操作,用于保持树的平衡。
  5. flipColors方法:颜色翻转,用于调整节点的颜色。
  6. put方法:插入操作,通过递归插入新节点,并在必要时进行旋转和颜色调整。
  7. get方法:查找操作,通过比较键值,递归查找目标节点。
  8. main方法:测试插入和查找操作。
运行结果
one
two
three

4.2 Python 实现示例:二叉查找树(BST)

下面的Python代码展示了一个简化的二叉查找树实现,包括插入、查找和中序遍历操作。

class TreeNode:def __init__(self, key, value):self.key = keyself.value = valueself.left = Noneself.right = Noneclass BinarySearchTree:def __init__(self):self.root = Nonedef get(self, key):return self._get(self.root, key)def _get(self, node, key):if node is None:return Noneif key < node.key:return self._get(node.left, key)elif key > node.key:return self._get(node.right, key)else:return node.valuedef put(self, key, value):if self.root is None:self.root = TreeNode(key, value)else:self._put(self.root, key, value)def _put(self, node, key, value):if key < node.key:if node.left is None:node.left = TreeNode(key, value)else:self._put(node.left, key, value)elif key > node.key:if node.right is None:node.right = TreeNode(key, value)else:self._put(node.right, key, value)else:node.value = valuedef inorder(self):return self._inorder(self.root)def _inorder(self, node):if node is None:return []return self._inorder(node.left) + [(node.key, node.value)] + self._inorder(node.right)# 测试
bst = BinarySearchTree()
bst.put(1, 'one')
bst.put(2, 'two')
bst.put(3, 'three')print(bst.get(1))  # 输出: one

五、结论

树表查找算法通过巧妙地利用树结构,实现了高效的查找、插入和删除操作。它们在数据库、内存数据结构和文件系统中有着广泛的应用。虽然实现复杂度较高,但其优越的性能和动态性使其成为处理大量数据的理想选择。通过本文的介绍和示例代码,希望你能够对树表查找算法有更深入的理解和应用。

下期见啦~

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

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

相关文章

【DPDK学习路径】一、前言及目录

虽然目前网络上已经有很多关于DPDK的帖子&#xff0c;DPDK官网也有自己的说明文档&#xff0c;但是这些现存的资料&#xff0c;要么不够系统、全面&#xff0c;要么入门门槛很高&#xff0c;需要非常精通操作系统及网络报文处理&#xff0c;甚至于要你本身已经对DPDK十分了解才…

复合机器人以其高度的灵活性和操作效率,展现了显著的优势

随着工业4.0的深入推进和智能制造的快速发展&#xff0c;复合机器人作为一种集成移动机器人和工业机器人功能的先进设备&#xff0c;正逐步成为工业自动化领域的新宠。特别是在磁钢上下料的应用中&#xff0c;复合机器人以其高度的灵活性和操作效率&#xff0c;展现了显著的优势…

基于C#开发web网页管理系统模板流程-主界面密码维护功能完善

点击返回目录-> 基于C#开发web网页管理系统模板流程-总集篇-CSDN博客 前言 紧接上篇->基于C#开发web网页管理系统模板流程-主界面统计功能完善-CSDN博客 一个合格的管理系统&#xff0c;至少一定存在一个功能——用户能够自己修改密码&#xff0c;理论上来说密码只能有用…

Matlab图像处理——细胞图像的分割和计数显示

一. 项目介绍 使用MATLAB编写的细胞图像分割及计数系统&#xff0c;实现了对图像内细胞的计数&#xff0c;以及对每个细胞周长和面积的测量&#xff0c;并分别展示了分割后的每个细胞的图像。实验步骤共分为图像预处理、图像预分割、空洞填充、黏连细胞分割、细胞个数统计、细胞…

【Python入门与进阶】Python中变量的输出方式

在Python中&#xff0c;有多种方式来输出变量的值。以下是几种常见的方法&#xff1a; 1. 使用 print() 函数 这是最基本和常用的输出方法。 x 10 print(x)2. 使用格式化字符串&#xff08;f-strings&#xff09; f-strings 是在 Python 3.6 引入的&#xff0c;它们非常方…

TikTok网红营销指南 | 怎么找到TikTok网红并进行合作?

如果你打算在tiktok上进行营销&#xff0c;忽略与tiktok网红合作无异于错失良机&#xff0c;时尚博主Sophia仅用一条30秒的视频展示了自己从一家新兴品牌购买的连衣裙&#xff0c;视频迅速获得了数百万的点赞和评论&#xff0c;也让该品牌的销量翻了好几倍。 这种与网红合作的策…

Qt绘图项目 - 简易表盘

废话少说&#xff0c;放码过来 widget.h #ifndef WIDGET_H #define WIDGET_H#include <QWidget>QT_BEGIN_NAMESPACE namespace Ui { class Widget; } QT_END_NAMESPACEclass Widget : public QWidget {Q_OBJECTpublic:Widget(QWidget *parent nullptr);~Widget();prot…

LayUI使用(二)处理表格会出现下拉框的问题

一、问题描述 如下&#xff0c;layui的表格渲染后&#xff0c;当鼠标悬停在表格项时会出现右侧的下拉框&#xff0c;layui版本较老&#xff0c;原因未知 二、处理办法 在cols里面加上width&#xff0c;也不用每个都加&#xff0c;加一部分表格项即可 注意&#xff1a;若想禁止…

iOS/iPadOS18Beta是否值得升级体验?Bug汇总和升级办法分享!

苹果昨天发布了iOS/iPadOS18Beta更新&#xff0c;引入了诸多新功能/新特性&#xff0c;很多喜欢尝鲜的用户已经在第一时间进行了升级。 iOS/iPadOS18Beta目前存在不少Bug&#xff0c;建议暂时不要更新&#xff0c;轻则浪费装机时间&#xff0c;重则丢失相关数据&#xff0c;甚至…

ping: www.baidu.com: 未知的名称或服务(IP号不匹配)

我用的是VMware上的Red Hat Enterprise Linux 9&#xff0c;出现了能联网但ping不通外网的情况。 问题描述&#xff1a;设置中显示正常连接&#xff0c;而且虚拟机右上角有联网的图标&#xff0c;但不能通外网。 按照网上教程修改了/etc/resolv.conf和/etc/sysconfig/network-…

江协科技51单片机学习-1 安装Keil5开发环境

前言&#xff1a; 本文是根据哔哩哔哩网站上“江协科技51单片机”视频的学习笔记&#xff0c;在这里会记录下江协科技51单片机开发板的配套视频教程所作的实验和学习笔记内容。本文大量引用了江协科技51单片机教学视频和链接中的内容。 引用&#xff1a; 51单片机入门教程-2…

Flask基础2-Jinja2模板

目录 1.介绍 2.模板传参 1.变量传参 2.表达式 3.控制语句 4.过滤器 5.自定义过滤器 6.测试器 7.块和继承 flask基础1 1.介绍 Jinja2:是Python的Web项目中被广泛应用的模板引擎,是由Python实现的模板语言,Jinja2 的作者也是 Flask 的作 者。他的设计思想来源于Django的模…

Shell脚本从入门到实战

一、概述 shell 是一个命令行解释器&#xff0c;它接受应用程序、用户命令&#xff0c;然后调用操作系统内核。 shell 还是一个功能强大编程语言&#xff0c;易调试&#xff0c;易编写&#xff0c;灵活性强。 二、mac 怎么重启docker 1.如何重启 Docker on Mac 在 macOS 上…

Python界面编辑器Tkinter布局助手 使用体验

一、发现 我今天在网上搜关于Python Tkinter方面的信息时&#xff0c;发现了Python界面编辑器 Tkinter布局助手 的使用说明。 https://blog.csdn.net/weixin_52777652/article/details/135291731?spm1001.2014.3001.5506 这个编辑器是个开源的项目&#xff0c;个人用户可以…

用winform开发一个笔记本电脑是否在充电的小工具

笔记本充电状态有两种监测方式&#xff0c;一种是主动查询&#xff0c;另一种是注册充电状态变化事件 1&#xff0c;先说主动监控吧&#xff0c;建立一个线程&#xff0c;反复查询SystemInformation.PowerStatus.PowerLineStatus private void readPower(){while (true){this.…

随手记:商品信息过多,展开收起功能

UI原型图&#xff1a; 页面思路&#xff1a; 在商品信息最小item外面有一个包裹所有item的标签&#xff0c;控制这个标签的高度来实现展开收起功能 <!-- 药品信息 --><view class"drugs" v-if"inquiryInfoSubmitBtn"><view class"…

C++ 14 之 宏函数

c14宏函数.cpp #include <iostream> using namespace std;// #define PI 3.14 // 宏函数 // 宏函数缺陷1: 必须用括号保证运算的完整性 #define MY_ADD(x,y) ((x)(y))// 宏函数缺陷2&#xff1a;即使加了括号&#xff0c;有些运算依然与预期不符 #define MY_COM(a,b) ((…

技术革新,智绘未来丨悦数图数据库 v5.0 重磅亮相 WAIC 2024

本次 WAIC&#xff08;世界人工智能大会&#xff09;2024 将于7 月 4 日- 7 日在上海世博展览馆**举行&#xff0c;本次 WAIC 2024 围绕“以共商促共享 以善治促善智”为主题&#xff0c;杭州悦数科技有限公司将携最新的悦数图数据库 v5.0 亮相 E805 展位。作为国内领先的图数据…

2024/06/13--代码随想录算法(贪心)3/6|134.加油站、135.分发糖果、860.柠檬水找零、406.根据身高重建队列

134.加油站 力扣链接 class Solution:def canCompleteCircuit(self, gas: List[int], cost: List[int]) -> int:curSum 0 # 当前累计的剩余油量totalSum 0 # 总剩余油量start 0 # 起始位置for i in range(len(gas)):curSum gas[i] - cost[i]totalSum gas[i] - co…

二十三、生成帮助文档

二十一、Java工具类的创建 二十二、Jar包制作及使用 这一篇开始学习如何生成帮助文档。为什么要学习生成帮助文档&#xff1f; 1、工具类已经制作好了&#xff0c;Java工具类的创建的类是一个.java文件&#xff0c;编译后成.class文件看不懂&#xff0c;所以需要对应的帮助文档…