算法与数据结构——哈希优化策略与算法选择

news/2024/9/19 14:13:17/文章来源:https://www.cnblogs.com/1873cy/p/18410045

哈希优化策略

在算法题中,我们通常通过线性查找替换为哈希查找来降低算法的时间复杂度。我们借助一个算法题来加深理解。

Question
给定一个整数数组nums和一个目标元素target,请在数组中搜索“和”为target的两个元素,并返回他们的数组索引。返回任意一个即可。

线性查找: 以时间换空间

考虑直接遍历所有可能的组合。如图所示,我们开启一个两层循环,在每轮中判断两个整数的和是否为target,若是,则返回它们的索引。

/* 方法一:暴力枚举 */
vector<int> twoSumBruteForce(vector<int> &nums, int target) {int size = nums.size();for(int i = 0; i < size; ++i) {for(int j = i + 1; j < size; ++j) {if(nums[i] + nums[j] == target)return {i,j};}}return {};
}

此方法的时间复杂度为O(n2),空间复杂度为O(1),在大数据量下非常耗时。

哈希查找: 以空间换时间

考虑借助一个哈希表,键值对分别为数组元素和元素索引。循环遍历数组,每轮执行如下步骤:

  • 判断数组target - nums[i]是否在哈希表中,若是,则直接返回这两个元素的索引。
  • 将键值对nums[i]和索引i添加进哈希表。
/* 方法二:辅助哈希表 */
vector<int> twoSumHashTable(vector<int> &nums, int target) {// 辅助哈希表,空间复杂度为 O(n)unordered_map<int,int> numsMap;for(int i = 0; i < nums.size(); ++i) {if(numsMap.find(target - nums[i]) != numsMap.end())return {numsMap[target - nums[i]],i};numsMap.emplace(make_pair(nums[i], i));}return {};
}

此方法通过哈希查找将时间复杂度从O(n2)降至O(n),大幅度提升运行效率。

由于需要维护一个额外的哈希表,因此空间复杂度为O(n)。尽管如此,该方法的整体时空效率更为均衡,因此它是本题的最优解

重识搜索算法

搜索算法(search algorithm)用于在数据结构(例如数组、链表、树或图)中搜索一个或一组满足特定条件的元素。

搜索算法可根据实现思路分为以下两类。

  • 通过遍历数据结构来定位目标元素,例如数组、链表、树和图的遍历等。
  • 利用数据组织结构或数据包含的先验信息,实现高效查找,例如二分查找、哈希查找和二叉搜索树查找等。

暴力搜索

暴力搜索通过遍历数据结构的每个元素来定位目标元素。

  • “线性搜索”适用于数组和链表等线性数据结构。它从数据结构的一端开始,逐个访问元素,直到找到目标元素或到达另一端仍没有找到目标元素为止。
  • “广度优先搜索”和“深度优先搜索”是图和树的两种遍历策略。广度优先搜索从初始节点开始逐层搜索,由近及远地访问各个节点。深度优先搜索从初始节点开始,沿着一条路径走到头,在回溯并尝试其他路径,直到遍历完整个数据结构。

暴力搜索的优点是简单且通用性好,无须对数据做预处理和借助额外的数据结构

但是此类算法的时间复杂度为O(n),其中n为元素数量,因此在数据量较大情况下,性能较差。

自适应搜索

自适应搜索利用数据的特有属性(例如有序性)来优化搜索过程,从而更高效地定位目标元素。

  • “二分查找”利用数据的有序性实现高效查找,仅适用于数组。
  • “哈希查找”利用哈希表将搜索数据和目标数据建立为键值对映射,从而实现查询操作。
  • “树查找”在特定的树结构(例如二叉搜索树)中,基于比较节点值来快速排除节点,从而定位目标元素。

此类算法的优点是效率高,时间复杂度可达到O(logn)甚至O(1)

在使用这些算法时往往需要对数据进行预处理。例如,二分查找需要预先对数组进行排序,哈希查找和树查找都需要借助额外的数据结构,维护这些数据结构也需要额外的时间和空间开销。

自适应搜索算法通常被称为查找算法,主要用于在特定数据结构中快速检索目标元素

搜索方法选取

给定大小为n的一组数据,我们可以使用线性搜索、二分查找、树查找、哈希查找等多种方法从中搜索目标元素。各个方法的工作原理如下图:

上述几种方法的操作效率与特性如下表所示:

线性搜索 二分查找 树查找 哈希查找
查找元素 O(n) O(logn) O(logn) O(1)
插入元素 O(1) O(n) O(logn) O(1)
删除元素 O(n) O(n) O(logn) O(1)
额外空间 O(1) O(1) O(n) O(n)
数据预处理 / 排序O(nlogn) 建树O(nlogn) 建哈希表O(n)
数据是否有序 无序 有序 有序 无序

搜索算法的选择还取决于数据体量、搜索性能要求、数据查询与更新频率等。

线性搜索

  • 通用性较好,无须任何数据预处理操作。加入我们仅需查询一次数据,那么其他三种方法的数据预处理的时间比线性搜索的时间还要更长。
  • 适用于体量较小的数据,此情况下时间复杂度对效率影响较小。
  • 适用与数据更新频率较高的场景,因为该方法不需要对数据进行任何额外维护。

二分查找

  • 适用于大数据量的情况,效率表现稳定,最差时间复杂度为O(logn)。
  • 数据量不能过大,因为存储数组需要连续的内存空间。
  • 不适用与高频增删数据的场景,因为维护有序数组的开销较大。

哈希查找

  • 适合对查询性能要求很高的场景,平均时间复杂度为O(1)。
  • 不适合需要有序数据或范围查找的场景,因为哈希表无法维护数据的有序性。
  • 对哈希函数和哈希冲突处理策略的依赖性较高,具有较大的性能劣化风险。
  • 不适合数据量过大的情况,因为哈希表需要额外空间来最大程度地减少冲突,从而提供良好的查询性能。

树查找

  • 适用于海量数据,因为树节点在内存中是分散存储的。
  • 适合需要维护有序数据或范围查找的场景。
  • 在持续增删节点过程中,二叉搜索树可能产生倾斜,时间复杂度劣化至O(n)。
  • 若使用AVL树或红黑树,则各项操作可在O(logn)效率下稳定运行 ,但维护树平衡的操作会增加额外的开销。

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

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

相关文章

织梦DEDECMS怎么实现全站动态浏览

要实现DedeCMS(织梦CMS)全站动态浏览,可以通过以下步骤来进行配置:首页动态化:登录织梦CMS的后台管理系统。 导航到“核心”->“全局配置”。 在“站点设置”标签页中,找到“主页网址”设置,确保主页网址是动态的,例如 http://www.example.com/ 而不是静态的 http:/…

织梦dedecms怎么调用图片集中图片的注释

在DedeCMS中调用图片集中的图片及其注释,可以通过自定义函数或者利用已有的函数来实现。下面是一个基于已有资料的示例,展示如何调用图片集中的图片及其注释。 首先,你需要确保你的图片已经被正确地添加到了织梦CMS的图集功能中。然后,你可以使用自定义函数来获取这些图片及…

JavaScript:对组织值进行排序

要对组织值进行排序,你可以使用JavaScript中的数组排序方法 sort()。下面是一些示例代码,展示如何对不同类型的组织值进行排序: 示例 1:对数字数组进行排序const numbers = [5, 2, 9, 1, 5, 6];// 使用 sort() 方法进行升序排序 numbers.sort((a, b) => a - b);console.…

dedecms缩略图报错怎么办

当遇到DedeCMS(织梦CMS)中缩略图报错的问题时,可以尝试以下几种解决方法来定位和解决问题:检查图片路径:确保缩略图的路径是正确的,有时候图片路径错误会导致缩略图无法显示。检查图片文件:确保图片文件本身没有损坏,并且是服务器支持的格式(如 .jpg, .png, .gif 等)…

如何优雅地处理返回值

我们已经知道了如何优雅的校验传入的参数了,那么后端服务器如何实现把数据返回给前端呢? 返回格式 后端返回给前端我们一般用 JSON 体方式,定义如下: {#返回状态码code:string, #返回信息描述message:string,#返回值data:object }CODE 状态码 Code 返回状态码,一般是…

白云龙期货分析-第九讲

九种职业操盘手高级战法 出征准备买卖法V旗买卖法双旗合并买卖法青龙取水买卖法中长线可选60日均线水杯买卖法南辕北辙买卖法国内和国外趋势不一样 断剑买卖法大阴阳买卖法童子拜佛买卖法 下影线多于上影线,向上趋势能量

浮点数的比较

浮点数与"零值" 精度损失: 浮点值与实际值不等,可能偏大可能偏小,都属于精度损失验证浮点数是否存在精度损失验证浮点数的差值是否存在精度损失浮点数直接比较验证结论: 浮点数在进行比较时,绝对不能使用双等号==来进行比较. 浮点数本身有精度损失,进而导致结果可能…

白家强的第一次作业

一、自我介绍 大家好,我的名字是白家强,我来自河北邢台,我的家乡处于华北平原,在我的家乡有美丽的苞米地和一望无垠的大麦田。我现就读于浙江理工大学22级自动化(1)班,是一名大三的学生。在杭州上学的这两年中,遇到了许许多多的人,结识了许多好友,认识了很多老师,我…

Qt加载天地图离线api开发包/从官网趴地图js代码/费了九牛二虎之力终于搞定

一、前言说明 网上关于如何趴天地图离线api文件的文章,只有少量的两三篇,而且几乎没有说全和说对,搞得评论也是一片懵逼,这里不行那你不行,思路可以借鉴就是。索性花了点时间,自己研究了如何从官网一步步趴下来js文件,最终所有离线能使用的功能全部搞定,也根本不会有ht…

一键生成!轻量级 AI 证件照制作工具!

HivisionIDPhotos —— 一个轻量级的 AI 证件照制作工具,利用一套完善的模型工作流程,实现对多种用户拍照场景的识别、抠图与证件照生成。大家好,我是 Java陈序员。 在日常生活中,我们需要各式各样的证件照。有时候需要不同的尺寸,一寸、两寸、小二寸...而有的时候需要不同…

[问题排查]CPU占用过高

查看占用cpu高的进程 top 按P 如下图:可看出PID为7149的java进程占用cpu最高,达到了98%查看进程中最耗cpu的子线程 top -Hp 7149 如下图:可看出PID为7166的线程占用cpu最高,达到了97.7%将最耗cpu的线程id转换为16进制输出 perl printf "%x \n" 7166 查询具体出现…

白云龙期货投资-第八讲

假突破的深度分析 一 高低点 平台之假突破深度分析 二 趋势线假突破之深度分析 三 假突破实盘应变操作法 假突破之飘旗确认法 在上沿做旗形整理,假突破假突破之空间确认法盘久必跌 趋势线假突破之深度分析假突破实盘应变方法 一 任何突破后不要急于进场,要看下一步行情的反应…