用go语言实现树和哈希表算法
算法复杂度
判断一个算法的效率通常基于其计算复杂度,这主要与算法访问输入数据的次数有关。计算机科学中常用大O表示法来描述算法的复杂度。例如,O(n)的算法只需访问一次输入数据,因此优于O(n²)的算法,后者则优于O(n³)的算法,依此类推。最差的算法是O(n!)的复杂度,当输入数据超过300个元素时,这样的算法几乎无法使用。
在Go语言中,大多数内建类型的查找操作(如通过键值查找map中的元素或访问数组元素)都具有常数时间复杂度,表示为O(1)。这意味着内建类型通常比自定义类型更快,除非你希望对底层行为进行完全控制,否则应优先选择使用内建类型。
不仅如此,不同的数据结构效率各不相同。通常,数组操作比map操作要快,但map的多功能性使它具有独特的优势。因此,开发者在选择数据结构时需权衡这些特性。
Go中的二叉树
二叉树简介
二叉树是一种数据结构,每个节点最多有两个子节点,即一个节点可以与最多两个其他节点相连。二叉树的根节点是树的第一个节点。树的深度(也称为高度)是从根节点到某个节点的最长路径,而某个节点的深度是该节点到根节点的边数。没有子节点的节点称为叶子节点。
当一棵树的最长路径与最短路径之间的差值不超过1时,称其为平衡树。如果不满足这一条件,则为不平衡树。树的平衡操作通常较为复杂且耗时,因此最好在树创建时保持其平衡,特别是在节点数量较多的情况下。
二叉树的实现
在Go中,二叉树的实现可以通过结构体定义节点。下面是实现一个简单二叉树的代码,并带有中文注释。
package mainimport ("fmt""math/rand""time"
)// 定义二叉树节点结构
type Tree struct {Left *Tree // 左子节点Value int // 节点的值Right *Tree // 右子节点
}// 遍历二叉树
func traverse(t *Tree) {if t == nil {return}traverse(t.Left) // 递归遍历左子树fmt.Print(t.Value, " ") // 打印当前节点的值traverse(t.Right) // 递归遍历右子树
}// 创建二叉树并填充随机值
func create(n int) *Tree {var t *Treerand.Seed(time.Now().Unix()) // 初始化随机数种子for i := 0; i < 2*n; i++ {temp := rand.Intn(n * 2)t = insert(t, temp) // 插入随机值}return t
}// 插入节点到二叉树中
func insert(t *Tree, v int) *Tree {if t == nil {return &Tree{nil, v, nil} // 创建根节点}if v == t.Value {return t // 如果值已存在,不做操作}if v < t.Value {t.Left = insert(t.Left, v) // 递归插入到左子树return t}t.Right = insert(t.Right, v) // 递归插入到右子树return t
}func main() {tree := create(10)fmt.Println("树的根节点值为:", tree.Value)traverse(tree)fmt.Println()// 插入新值并再次遍历tree = insert(tree, -10)tree = insert(tree, -2)traverse(tree)fmt.Println()fmt.Println("树的根节点值为:", tree.Value)
}
运行结果:
树的根节点值为: 18
0 3 4 5 7 8 9 10 11 14 16 17 18 19
-10 -2 0 3 4 5 7 8 9 10 11 14 16 17 18 19
树的根节点值为: 18
二叉树的优势
二叉树特别适合用于表示层次结构的数据,因此在编译器解析程序代码时,广泛采用二叉树。此外,二叉树是天然有序的,只需插入元素到正确位置,树结构就会保持有序。然而,删除树中的元素相对复杂,因为需要维护树的结构。
当二叉树是平衡的,其查找、插入和删除操作的时间复杂度大约为O(log n),其中n是树中元素的数量。例如,一个包含100万个元素的平衡树,其高度大约为20,这意味着可以在不到20步内访问到树中的任意节点。
二叉树的主要缺点在于其结构取决于插入元素的顺序。如果树的键值较长且复杂,插入和查找操作可能会变慢。此外,如果树不平衡,树的性能将变得不可预测。
哈希表在Go中的应用
哈希表的概念
哈希表是一种存储键值对的数据结构,它通过哈希函数计算出一个索引,从而定位数据。一个好的哈希函数需要能够产生均匀分布的哈希值,以避免哈希冲突。
Go中的哈希表实现
下面展示了如何在Go中实现一个简单的哈希表:
package mainimport ("fmt"
)// 定义哈希表的大小
const SIZE = 15// 定义哈希表的节点结构
type Node struct {Value intNext *Node
}// 定义哈希表结构
type HashTable struct {Table map[int]*NodeSize int
}// 哈希函数
func hashFunction(i, size int) int {return i % size
}// 插入数据到哈希表
func insert(hash *HashTable, value int) int {index := hashFunction(value, hash.Size)element := Node{Value: value, Next: hash.Table[index]}hash.Table[index] = &elementreturn index
}// 遍历哈希表
func traverse(hash *HashTable) {for k := range hash.Table {if hash.Table[k] != nil {t := hash.Table[k]for t != nil {fmt.Printf("%d -> ", t.Value)t = t.Next}fmt.Println()}}
}func main() {// 创建哈希表table := make(map[int]*Node, SIZE)hash := &HashTable{Table: table, Size: SIZE}fmt.Println("哈希表的大小为:", hash.Size)// 向哈希表插入数据for i := 0; i < 120; i++ {insert(hash, i)}// 遍历并打印哈希表traverse(hash)
}
运行结果:
哈希表的大小为: 15
105 -> 90 -> 75 -> 60 -> 45 -> 30 -> 15 -> 0 ->
110 -> 95 -> 80 -> 65 -> 50 -> 35 -> 20 -> 5 ->
...
哈希表的优势
哈希表的最大优势在于查找速度快。当哈希表有n个键和k个桶时,查找时间复杂度从O(n)降低到O(n/k),即使哈希表中有大量元素,查找效率也能保持在较低的时间复杂度内。
补充知识点
-
二叉树的平衡与自平衡树:虽然普通二叉树的性能取决于插入顺序,但一些自平衡树(如AVL树和红黑树)通过自动调整树的结构,确保即使在最差情况下也能维持较优的性能。
-
哈希碰撞处理:在哈希表中,多个键可能会映射到同一个索引,这被称为哈希碰撞。常用的碰撞处理方法有链地址法和开放地址法。在链地址法中,每个桶包含一个链表,用于存储冲突的键值对。
通过本文的讲解,相信大家对数据结构如二叉树和哈希表在Go中的应用有了更深入的理解。掌握这些基础结构,不仅能提升代码效率,还能为复杂项目的实现打下坚实的基础。
相关文章:
用go语言实现树和哈希表算法
算法复杂度 判断一个算法的效率通常基于其计算复杂度,这主要与算法访问输入数据的次数有关。计算机科学中常用大O表示法来描述算法的复杂度。例如,O(n)的算法只需访问一次输入数据,因此优于O(n)的算法,后者则优于O(n)的算法&…...

基于SpringBoot+Vue+MySQL的校园健康驿站管理系统
系统展示 用户前台界面 管理员后台界面 系统背景 本文设计并实现了一个基于SpringBoot后端、Vue前端与MySQL数据库的校园健康驿站管理系统。该系统旨在通过数字化手段,全面管理学生的健康信息,包括体温监测、疫苗接种记录、健康状况申报等,为…...
深入理解MATLAB中的事件处理机制
在MATLAB中,事件处理机制是一种强大的工具,它允许对象之间的交互和通信。这种机制基于观察者设计模式,其中一个对象(观察者)监听另一个对象(发布者)的状态变化。当发布者的状态发生变化时&#…...

线程--线程同步
这里写目录标题 同步概念线程同步概念数据混乱原因 互斥量原理锁的注意事项1、cpu时间轮片2、建议锁总结 使用锁来管理线程同步问题产生主要函数init、destorylock、unlock代码注意事项(锁的粒度) try锁死锁出现原因图解 读写锁特性图解函数总览init、de…...

【QT】Qt窗口
欢迎来到Cefler的博客😁 🕌博客主页:折纸花满衣 🏠个人专栏:QT 目录 👉🏻菜单栏设置👉🏻QToolBar练习 👉🏻QStausBar👉🏻Q…...

场外个股期权怎么给股票加杠杆?
今天期权懂带你了解场外个股期权怎么给股票加杠杆?场外期权交易通过向证券公司支付一定额度的股票期权费,然后买入大额的股票持仓,从而实现的杠杆交易。 买入看涨期权 操作:支付权利金购买看涨期权。 杠杆作用: 期…...

【Docker部署ELK】(7.15)
1、拉取镜像 docker pull docker.elastic.co/elasticsearch/elasticsearch:7.15.0 docker pull docker.elastic.co/kibana/kibana:7.15.0 docker pull docker.elastic.co/logstash/logstash:7.15.02、配置文件(解压资源到D盘DOCKER目录下) 2.1 配置文件…...

UE4_后期处理_后期处理材质及后期处理体积一
后期处理效果 在渲染之前应用于整个渲染场景的效果。 后期处理效果(Post-processing effect)使美术师和设计师能够对影响颜色、色调映射、光照的属性和功能进行组合选择,从而定义场景的整体外观。要访问这些功能,可以将一种称为…...

【PyQt6 应用程序】基于QtDesigner做一个用户登录页面
在当今的软件开发领域,用户界面(UI)设计和后端编程是创建现代、互动应用程序的两大重要组成部分。尤其是在开发具有用户登录功能的应用程序时,不仅要注重外观和用户体验的设计,还要确保后端逻辑的安全性和可靠性。 本文将介绍如何使用PyQt6框架结合UI设计,实现一个简单而…...

Ollama—87.4k star 的开源大模型服务框架!!
这一年来,AI 发展的越来越快,大模型使用的门槛也越来越低,每个人都可以在自己的本地运行大模型。今天再给大家介绍一个最厉害的开源大模型服务框架——ollama。 项目介绍 Ollama 是一个开源的大语言模型(LLM)服务工具…...

MySQL表的操作与数据类型
目录 前言 一、表的操作 1.创建一个表 2.查看表的结构 3.修改表 4.删除一个表 二、 MySQL的数据类型 0.数据类型一览: 1.整数类型 2.位类型 3.小数类型 4.字符类型 前言 在MySQL库的操作一文中介绍了有关MySQL库的操作,本节要讲解的是由库管理的结构——…...
mysql把某一个字段的值中的aa,替换成bb
UPDATE my_table SET my_column REPLACE(my_column, aa, bb); 例 假设my_table表在替换前的数据如下: idmy_column1hello aa2world aa aa3no aa here 执行上述UPDATE语句后,my_table表的数据将变为: idmy_column1hello bb2world bb b…...
【系统架构设计师】原型模式详解
原型模式详解 1. 什么是原型模式? 原型模式(Prototype Pattern)是一种创建型设计模式,它允许通过复制已有的对象来创建新的对象,而不是通过类实例化来创建新对象。通过这种方式,原型模式能够减少创建对象的开销,尤其是当对象的创建过程非常复杂或者耗费资源时。原型模…...
Spring @Async 深度解读:默认线程池执行器的配置与优化
在Spring中,Async注解用于异步执行方法。默认情况下,Async注解的任务是由一个线程池执行的。然而,这个默认的线程池是如何初始化的呢?本文将深入探讨这一过程,帮助你理解Spring异步任务背后的线程池执行器的初始化原理…...

手把手教你用护核纪元地心护核者用服务器开服联机
1、购买后登录服务器面板(百度莱卡云面板) 登录面板的信息在绿色的登陆面板按键下方,不是你的莱卡云账号 进入控制面板后会出现正在安装的界面,安装大约3分钟(如长时间处于安装中请联系我们的客服人员) 2、…...
Log4j 1.x如何升级到Log4j 2.x
Log4j 1.x升级到Log4j 2.x是一个涉及多个步骤的过程,主要包括删除旧版本、添加新版本依赖、配置新版本的配置文件等。以下是一个详细的升级步骤指南: 一、准备阶段 了解当前项目依赖: 检查项目中所有使用Log4j 1.x的地方,包括ja…...

CloudFlare问题与CDN问题
昨天将腾讯云的解析转移到Cloudflare中了,结果今天发现网站崩了,显示重定向次数过多,昨天估计是因为浏览器缓存,所以没有发现问题 问题一:强制HTTPS 当时看到CloudFlare的强制https时就想到了我的宝塔面板也开着强制h…...

[Linux]:文件(上)
✨✨ 欢迎大家来到贝蒂大讲堂✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 所属专栏:Linux学习 贝蒂的主页:Betty’s blog 1. C语言文件操作 C语言文件操作接口如下,详情可参照——C语言文…...
flutter开发多端平台应用的探索 下 (跨模块、跨语言通信之平台通道)
前文 Flutter 是一个跨平台的开发框架,它允许开发者使用相同的代码库来构建 iOS、Android、Web 和桌面应用程序。 上文flutter开发多端平台应用的探索 上(基本操作)-CSDN博客列举了一些特定平台的case(桌面端菜单,鼠…...

第15-02章:理解Class类并获取Class实例
我的后端学习大纲 我的Java学习大纲 1、Java反射机制原理图: 源代码通过Javac编译得到字节码文件,当我执行到new一个对象的时候,字节码文件会通过ClassLoader被加载,然后得到一个Class类对象,存放在堆中,加…...

网络编程(Modbus进阶)
思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…...
Python|GIF 解析与构建(5):手搓截屏和帧率控制
目录 Python|GIF 解析与构建(5):手搓截屏和帧率控制 一、引言 二、技术实现:手搓截屏模块 2.1 核心原理 2.2 代码解析:ScreenshotData类 2.2.1 截图函数:capture_screen 三、技术实现&…...

微信小程序之bind和catch
这两个呢,都是绑定事件用的,具体使用有些小区别。 官方文档: 事件冒泡处理不同 bind:绑定的事件会向上冒泡,即触发当前组件的事件后,还会继续触发父组件的相同事件。例如,有一个子视图绑定了b…...
DeepSeek 赋能智慧能源:微电网优化调度的智能革新路径
目录 一、智慧能源微电网优化调度概述1.1 智慧能源微电网概念1.2 优化调度的重要性1.3 目前面临的挑战 二、DeepSeek 技术探秘2.1 DeepSeek 技术原理2.2 DeepSeek 独特优势2.3 DeepSeek 在 AI 领域地位 三、DeepSeek 在微电网优化调度中的应用剖析3.1 数据处理与分析3.2 预测与…...

Nuxt.js 中的路由配置详解
Nuxt.js 通过其内置的路由系统简化了应用的路由配置,使得开发者可以轻松地管理页面导航和 URL 结构。路由配置主要涉及页面组件的组织、动态路由的设置以及路由元信息的配置。 自动路由生成 Nuxt.js 会根据 pages 目录下的文件结构自动生成路由配置。每个文件都会对…...
【android bluetooth 框架分析 04】【bt-framework 层详解 1】【BluetoothProperties介绍】
1. BluetoothProperties介绍 libsysprop/srcs/android/sysprop/BluetoothProperties.sysprop BluetoothProperties.sysprop 是 Android AOSP 中的一种 系统属性定义文件(System Property Definition File),用于声明和管理 Bluetooth 模块相…...

成都鼎讯硬核科技!雷达目标与干扰模拟器,以卓越性能制胜电磁频谱战
在现代战争中,电磁频谱已成为继陆、海、空、天之后的 “第五维战场”,雷达作为电磁频谱领域的关键装备,其干扰与抗干扰能力的较量,直接影响着战争的胜负走向。由成都鼎讯科技匠心打造的雷达目标与干扰模拟器,凭借数字射…...
根据万维钢·精英日课6的内容,使用AI(2025)可以参考以下方法:
根据万维钢精英日课6的内容,使用AI(2025)可以参考以下方法: 四个洞见 模型已经比人聪明:以ChatGPT o3为代表的AI非常强大,能运用高级理论解释道理、引用最新学术论文,生成对顶尖科学家都有用的…...

Maven 概述、安装、配置、仓库、私服详解
目录 1、Maven 概述 1.1 Maven 的定义 1.2 Maven 解决的问题 1.3 Maven 的核心特性与优势 2、Maven 安装 2.1 下载 Maven 2.2 安装配置 Maven 2.3 测试安装 2.4 修改 Maven 本地仓库的默认路径 3、Maven 配置 3.1 配置本地仓库 3.2 配置 JDK 3.3 IDEA 配置本地 Ma…...
基于Java Swing的电子通讯录设计与实现:附系统托盘功能代码详解
JAVASQL电子通讯录带系统托盘 一、系统概述 本电子通讯录系统采用Java Swing开发桌面应用,结合SQLite数据库实现联系人管理功能,并集成系统托盘功能提升用户体验。系统支持联系人的增删改查、分组管理、搜索过滤等功能,同时可以最小化到系统…...