【算法学习】哈希表篇:哈希表的使用场景和使用方法

算法学习:

https://blog.csdn.net/2301_80220607/category_12922080.html?spm=1001.2014.3001.5482

前言:

在之前学习数据结构时我们就学习了哈希表的使用方法,这里我们主要是针对哈希表的做题方法进行讲解,都是leetcode上的经典题,各位可以自己做一遍再来看一下,主要抽取了几个经典的题

目录

1. 哈希表的基础知识

1.1 哈希表是什么

1.2 哈希表的作用

1.3 什么时候用哈希表

1.4 哈希表的应用场景

2. 哈希表经典例题

2.1 两数之和

2.2 存在重复元素||

2.3 字母异位词分组

3. 总结


1. 哈希表的基础知识

1.1 哈希表是什么

简单点来说哈希表就是一个存储数据的容器,但是能够对出现的数据进行标记

1.2 哈希表的作用

能够实现快速查找指定的数据,时间复杂度往往为O(1)

1.3 什么时候用哈希表

频繁的查找数据时

1.4 哈希表的应用场景

1. 直接使用哈希表这个数据类型

2. 在合适的场景使用数组来模拟哈希表

解释:

2、3:当我们需要频繁查找数据的时候我们就可以想到要用哈希表,哈希表的时间复杂度为o(1),空间复杂度为o(n)同时也要考虑是否可以用二分,二分时间复杂度为o(logn),也非常快,而且二分没有时间开销
4:除了之间使用函数提供给我们的哈希表外,我们也可以用数组模拟创建一个简易的哈希表,但这种只适用于元素少的情况,比如让我们统计字符串中各字符出现次数,我们就可以用数组 int hash[26]模拟一下

2. 哈希表经典例题

2.1 两数之和

1. 两数之和

给定一个整数数组 nums 和一个整数目标值 target,请你在该数组中找出 和为目标值 target  的那 两个 整数,并返回它们的数组下标。

你可以假设每种输入只会对应一个答案,并且你不能使用两次相同的元素。

你可以按任意顺序返回答案。

示例 1:

输入:nums = [2,7,11,15], target = 9
输出:[0,1]
解释:因为 nums[0] + nums[1] == 9 ,返回 [0, 1] 。

示例 2:

输入:nums = [3,2,4], target = 6
输出:[1,2]

示例 3:

输入:nums = [3,3], target = 6
输出:[0,1]

提示:

  • 2 <= nums.length <= 104
  • -109 <= nums[i] <= 109
  • -109 <= target <= 109
  • 只会存在一个有效答案

算法原理:

解法一:暴力枚举

在这个暴力枚举中我们来考虑一种新的枚举思路

比如本题,按照之前的枚举策略,我们是要让当前位置(cur)与后面位置的数字依次相加,然后看是否有满足条件的数字

现在我们来看一种新的枚举思路,就是让所有位置的数字,与它前面位置的数字相加,然后判断是否有满足条件的两个数字这样的思路也能保证让所有的数两两相加,而且在处理一些细节上更有优势,尤其是本题,下面我们结合本题来感受一下

比如上面我们提到的这个队列,按照原来的思路,我们先来找一下2后面是否有合适的数字,使得2与其相加等于target,为了减小时间复杂度我们这里要用哈希表的方法,哈希表中存的值是数组中元素和对应的下标<nums[i,i>,但是如果我们是找当前元素后面的值时,我们就需要先提前将数组中所有数字及它们对应的下标先放入哈希表中,但这就会导致一些问题

比如这样一个数组和target值,如果我们按照原策略先将所有数字和对应下标放入哈希表中的话因为有两个3,在第一个3及它对应的下标放入哈希表中后,当放第二个3及它对应的下标时,就会把前一个存的3的下标给覆盖掉,所以我们还需要额外的操作对重复的数字的多个下标进行保存

但是如果我们采取的策略二,我们创建哈希表的时机就会发生一些变化,比如我们现在位置在2,我们是要从前面的位置找合适的数,前面没有数,所以哈希表中自然也没有,然后我们可以将2的值及它的下标放入哈希表中,同时让cur++,这样做的优化点是什么呢?

此时假如我们再遇到上面的情况(两个3),我们在第一个3时査找它前面是否有合适的数(前面的数己经放入哈希表了)时,会发现没有,然后我们把它放入哈希表中,然后到第二个3,我们发现哈希表中hash[target-3]存有一个下标,我们就找到满足条件的下标了

假如我们现在的target不是6,而是14,那我们在经过第二个3后,虽然会把哈希表中3对应的下标替换掉,但是仍不影响最终结果

思考一下:其实第二种策略比第一种策略的优化之处就是,策略二在替换一个数的坐标之前,是事先知道这个数字两两相加不满足条件的,所以保留一个与其它数字相加即可,而第一个数字则是在没考虑这种情况之前就只把重复数字只留下一个

代码实现:

class Solution {
public:vector<int> twoSum(vector<int>& nums, int target) {unordered_map<int,int> hash;for(int i=0;i<nums.size();i++){if(hash.count(target-nums[i])) return {hash[target-nums[i]],i};hash[nums[i]]=i;}return {-1,-1};}
};

2.2 存在重复元素||

219. 存在重复元素 II

给你一个整数数组 nums 和一个整数 k ,判断数组中是否存在两个 不同的索引 i 和 j ,满足 nums[i] == nums[j] 且 abs(i - j) <= k 。如果存在,返回 true ;否则,返回 false 。

示例 1:

输入:nums = [1,2,3,1], k = 3
输出:true

示例 2:

输入:nums = [1,0,1,1], k = 1
输出:true

示例 3:

输入:nums = [1,2,3,1,2,3], k = 2
输出:false

提示:

  • 1 <= nums.length <= 105
  • -109 <= nums[i] <= 109
  • 0 <= k <= 105

这道题就可以用上面的思路来解,挺简单的,可以练练手

代码实现:

class Solution {
public:bool containsNearbyDuplicate(vector<int>& nums, int k) {unordered_map<int,int> hash;for(int i=0;i<nums.size();i++){if(hash.count(nums[i])&&i-hash[nums[i]]<=k) return true;hash[nums[i]]=i;}return false;}
};

2.3 字母异位词分组

49. 字母异位词分组

给你一个字符串数组,请你将 字母异位词 组合在一起。可以按任意顺序返回结果列表。

字母异位词 是由重新排列源单词的所有字母得到的一个新单词。

示例 1:

输入: strs = ["eat", "tea", "tan", "ate", "nat", "bat"]
输出: [["bat"],["nat","tan"],["ate","eat","tea"]]

示例 2:

输入: strs = [""]输出: [[""]]

示例 3:

输入: strs = ["a"]输出: [["a"]]

提示:

  • 1 <= strs.length <= 104
  • 0 <= strs[i].length <= 100
  • strs[i] 仅包含小写字母

算法原理:

解释:

这道题就很考验我们对哈希表的使用,我们在这里创建的哈希表类型是一个<string,vector<string>>类型的哈希表,我们要做的是把所有的字母异位词放在一起,那么我们首先就需要判断哪些词是字母异位词,在这里我们有一个很巧的方法去判断,我们可以把这些字符串排序,字母异位词排序后是相等的,就如左边那样(“eat",“tea“排序后都是“aet"),然后让排序后的字符串作为key值,字母异位词放在同一个key值后,最后再从哈希表中把这些字符串数组提取出来即可

代码实现:

class Solution {
public:vector<vector<string>> groupAnagrams(vector<string>& strs) {unordered_map<string,vector<string>> hash;for(int i=0;i<strs.size();i++){string tmp=strs[i];sort(strs[i].begin(),strs[i].end());hash[strs[i]].push_back(tmp);}vector<vector<string>> ret;for(auto& [x,y]:hash)ret.push_back(y);return ret;}
};

3. 总结

以上就是几个关于哈希表的经典例题,都不算难,哈希表在算法题里出现的比例还是非常高的,一般都是作为一种数据类型存在的,掌握还是很有必要的

本篇笔记:

感谢各位大佬观看,创作不易,还望各位大佬点赞支持!!

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

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

相关文章

Java 中如何实现自定义类加载器,应用场景是什么?

在 Java 中&#xff0c;可以通过继承 java.lang.ClassLoader 类来实现自定义类加载器。自定义类加载器可以控制类的加载方式&#xff0c;实现一些特殊的应用场景。 实现自定义类加载器的步骤&#xff1a; 继承 java.lang.ClassLoader 类。 重写 findClass(String name) 方法 …

信创开发中跨平台开发框架的选择与实践指南

&#x1f9d1; 博主简介&#xff1a;CSDN博客专家、CSDN平台优质创作者&#xff0c;高级开发工程师&#xff0c;数学专业&#xff0c;10年以上C/C, C#, Java等多种编程语言开发经验&#xff0c;拥有高级工程师证书&#xff1b;擅长C/C、C#等开发语言&#xff0c;熟悉Java常用开…

WebRTC 服务器之Janus架构分析

1. Webrtc三种类型通信架构 1.1 1 对 1 通信 1 对 1 通信模型设计的主要⽬标是尽量让两个终端进⾏直联&#xff0c;这样即可以节省服务器的资源&#xff0c;⼜可以提⾼ ⾳视频的服务质量。WebRTC ⾸先尝试两个终端之间是否可以通过 P2P 直接进⾏通信&#xff0c;如果⽆法直接…

数字化转型进阶:26页华为数字化转型实践分享【附全文阅读】

本文分享了华为数字化转型的实践经验和体会。华为通过数字化变革,致力于在客户服务、供应链、产品管理等方面提高效率,并把数字世界带入每个组织,构建万物互联的智能世界。华为的数字化转型愿景是成为行业标杆,通过推进数字化战略、构建面向业务数字化转型的IT组织阵型、坚…

Hal库下备份寄存器

首先要确保有外部电源给VBAT供电 生成后应该会有这两个文件&#xff08;不知道为什么生成了好几次都没有&#xff0c;复制工程在试一次就有了&#xff09; 可以看到stm32f407有20个备份寄存器 读写函数 void HAL_RTCEx_BKUPWrite(RTC_HandleTypeDef *hrtc, uint32_t Backup…

使用 Vue3 + Webpack 和 Vue3 + Vite 实现微前端架构(基于 Qiankun)

在现代前端开发中&#xff0c;微前端架构逐渐成为一种流行的解决方案&#xff0c;尤其是在大型项目中。通过微前端&#xff0c;我们可以将一个复杂的单体应用拆分为多个独立的小型应用&#xff0c;每个子应用可以独立开发、部署和运行&#xff0c;同时共享主应用的基础设施。本…

【c++】【STL】list详解

目录 list的作用list的接口构造函数赋值运算符重载迭代器相关sizeemptyfrontbackassignpush_frontpop_frontpush_backpop_backinserteraseswapresizeclearspliceremoveremove_ifuniquemergesortreverse关系运算符重载&#xff08;非成员函数&#xff09; list的模拟实现结点类迭…

Redis持久化:

什么是Redis持久化&#xff1a; Redis 持久化是指将 Redis 内存中的数据保存到硬盘等持久化存储介质中&#xff0c;以便在 Redis 服务器重启或出现故障时能够恢复数据&#xff0c;保证数据的可靠性和持续性。Redis 提供了两种主要的持久化方式&#xff1a;RDB&#xff08;Redi…

VBA 64位API声明语句第009讲

跟我学VBA&#xff0c;我这里专注VBA, 授人以渔。我98年开始&#xff0c;从源码接触VBA已经20余年了&#xff0c;随着年龄的增长&#xff0c;越来越觉得有必要把这项技能传递给需要这项技术的职场人员。希望职场和数据打交道的朋友&#xff0c;都来学习VBA,利用VBA,起码可以提高…

在pycharm profession 2020.3将.py程序使用pyinstaller打包成exe

一、安装pyinstaller 在pycharm的项目的Terminal中运行pip3 install pyinstaller即可。 安装后在Terminal中输入pip3 list看一下是否成功 二、务必在在项目的Terminal中输入命令打包&#xff0c;命令如下&#xff1a; python3 -m PyInstaller --noconsole --onefile xxx.py …

Unity SpriteRenderer(精灵渲染器)

&#x1f3c6; 个人愚见&#xff0c;没事写写笔记 &#x1f3c6;《博客内容》&#xff1a;Unity3D开发内容 &#x1f3c6;&#x1f389;欢迎 &#x1f44d;点赞✍评论⭐收藏 &#x1f50e;SpriteRenderer:精灵渲染器 &#x1f4a1;Sprite Renderer是精灵渲染器&#xff0c;所有…

2.LED灯的控制和按键检测

目录 STM32F103的GPIO口 GPIO口的作用 GPIO口的工作模式 input输入检测 -- 向内检测 output控制输出 -- 向外输出 寄存器 寄存器地址的确定 配置GPIO口的工作模式 时钟的开启和关闭 软件编程驱动 LED 灯 硬件 软件 软件编程驱动 KEY 按键 硬件 软件 按键消抖 代码 STM32F…

Flink 的状态机制

在实时流处理领域&#xff0c;状态管理是构建复杂业务逻辑的核心能力。Apache Flink 通过统一的状态抽象和高效的容错机制&#xff0c;为开发者提供了从毫秒级窗口聚合到 TB 级历史数据关联的全场景支持。本文将深入剖析 Flink 状态机制的底层原理&#xff0c;结合实际案例展示…

【查看.ipynp 文件】

目录 如何打开 .ipynb 文件&#xff1f; 如果确实是 .ipynp 文件&#xff1a; .ipynp 并不是常见的 Jupyter Notebook 文件格式。通常&#xff0c;Jupyter Notebook 文件的扩展名是 .ipynb&#xff08;即 Interactive Python Notebook&#xff09;。如果你遇到的是 .ipynb 文…

Runnable组件重试机制降低程序错误率

一、LangChain 重试机制深度解析 当构建生产级AI应用时&#xff0c;with_retry() 机制可有效提升系统容错性&#xff0c;典型应用场景包括&#xff1a; API调用频率限制时的自动恢复模型服务临时不可用的故障转移网络波动导致的瞬时异常处理 参数详解与配置策略 1. 参数配置…

k8s笔记——kubebuilder工作流程

kubebuilder工作流程 Kubebuilder 工作流程详解 Kubebuilder 是 Kubernetes 官方推荐的 Operator 开发框架&#xff0c;用于构建基于 Custom Resource Definitions (CRD) 的控制器。以下是其核心工作流程的完整说明&#xff1a; 1. 初始化项目 # 创建项目目录 mkdir my-opera…

Java框架“若依RuoYi”前后端分离部署

运行环境 Eclipse IDE for Enterprise Java and Web Developers 下载Eclipse解压Eclipse到文件夹 Maven 下载Maven解压Maven到文件夹配置环境变量MAVEN_HOME为Maven安装位置配置环境变量path为%MAVEN_HOME%\bin Redis 下载Redis解压Redis到文件夹配置环境变量path为Redis安装位…

游戏引擎学习第249天:清理调试宏

欢迎大家&#xff0c;让我们直接进入调试代码的改进工作 接下来&#xff0c;我们来看一下上次停留的位置。如果我没记错的话&#xff0c;上一场直播的结尾我有提到一些我想做的事情&#xff0c;并且在代码中留下了一个待办事项。所以也许我们今天首先做的就是解决这个问题。但…

二极管反向恢复的定义和原理

二极管的反向恢复定义 二极管的反向恢复是指二极管从正向导通状态切换到反向阻断状态时&#xff0c;电流从正向变为负向并最终回到零所需的时间。具体过程如下&#xff1a; 正向导通&#xff1a;当二极管正向偏置时&#xff0c;电流可以顺利通过&#xff0c;此时二极管处于导…

音视频开发技术总结报告

音视频开发技术总结报告 一、音视频开发基础 1、音频基础 声音原理 声波特性&#xff1a;频率、振幅、波长人耳听觉范围&#xff1a;20Hz-20kHz声音三要素&#xff1a;音调、音量、音色 数字音频基础 采样率&#xff1a;常见44.1kHz、48kHz、96kHz量化位数&#xff1a;8bit、…