[Lc10_hash] 总结 | 两数之和 | 字符重排 | 存在重复元素 i ii | 字母异位词分组

news2025/4/21 23:50:47

目录

1.介绍

2.两数之和

题解

3.面试题 01.02. 判定是否互为字符重排

题解

4.存在重复元素

题解

5.存在重复元素 II

题解

⭕6.字母异位词分组

题解


1.介绍

哈希表是什么?

  • 存储数据的容器
  • 前文:[C++_] set | map | unordered_map

有什么用呢?

  • “快速” 查找某个元素。 时间复杂度O(1)

什么时候用哈希表?

频繁的查找某一个数的时候。

  • 频繁查找某一个数的时候,我们还要想到一个二分查找也可以快速查找某一个元素
  • 但是二分因为有些局限,具有二段性才可以用二分,它也比较快,时间复杂度O(logn)。
  • 而且能用二分我们尽量用二分,因为哈希表虽然非常快,但是它空间复杂度是O(n)。

怎么用哈希表?

  • 容器(哈希表)
  • 数组模拟简易哈希表

什么时候会想到用数组模拟简易哈希表,比如说:

  • 1. 关注字符串中的 “字符”

其中key就是index,value就是nums[index]

字符的ascll值当作index(key),nums[index]就是我们需要的值。非常快。

  • 2. 数据范围很小的时候

int 1~10^7, 最好不要有负数, - 10^3 ~ 10^3


2.两数之和

题目链接: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 不能选两个3。

解法一:暴力求解

  • 两层for循环把所有情况都找到,挑选符合条件的。
  • 以前我们是固定一个数,往后找。
class Solution {
public:
    vector<int> twoSum(vector<int>& nums, int target) 
    {
        for(int i=0;i<nums.size();i++)
        {
            for(int j=i+1;j<nums.size();j++)
            {
                if(nums[i]+nums[j]==target)
                    return {i,j};
            }
        }
        return {};
        
    }
};

但是我们还有一种暴力方法,固定一个数,往前找

  • 先固定其中一个数
  • 依次与该数之前的数相加

同样也能把所有情况都找完,并且少了判断边界情况。

解法二:利用哈希表做优化

  • 我们先看为什么暴力枚举过程这么慢。慢就慢在当固定一个数的时候
  • 比如说11,我们想在它的前面找到一个 target - nums[i] 的数 如 9 - 11 = - 2
  • 我们暴力策略就是从11之前依次遍历找这个-2。
  • 那此时我如果当固定一个数的时候就把它之前的是插入到hash表中,然后在hash表中就可以用O(1)的时间复杂度来找这个-2。

之前暴力解法O(N^2),利用哈希表做优化之后时间复杂度降到O(N)。

不过我们空间复杂度是O(N)的。

  • 具体如何操作可以和刚才的固定一个数往前找完美结合起来。
  • 当固定一个数的时候,就去hash表中找 target - nums[i] 的数在不在hash表。
  • 有就找到了返回对应两个数的下标即可,所有hash<nums[i] , i> 前面存值后面存对应的下标
  • 如果没找到就把这个数 加入到hash表中。

接下来扩展一下,为什么之前固定数,往后走不太好用呢?

  • 首先要将所有的数,先加入到hash表中,然后才能固定一个数,在去hash找 target - nums[i] 的数在不在hash表。
  • 注意我们这道题有个要求:数组中同一个数不能出现两次,如果target = 6,但是数组中有一个3, 你固定3然后在去hash表中找,找到的还是这个3,这是不对的。
  • 因为必须要加个if判断一下,当两个数相等时是不对的。
  • 定一个数往前找,是不会出现这种情况的。我们是先去找。没找到在把这个数加入到hash表中。

代码:

  • hash.count()
  • hash[num[i]]=i; //建立下标映射

3.面试题 01.02. 判定是否互为字符重排

题目链接:面试题 01.02. 判定是否互为字符重排

题目描述:

给定两个由小写字母组成的字符串 s1s2,请编写一个程序,确定其中一个字符串的字符重新排列后,能否变成另一个字符串。

示例 1:

输入: s1 = "abc", s2 = "bca"
输出: true

示例 2:

输入: s1 = "abc", s2 = "bad"
输出: false

题解

  • 可以把 “abc” 全排列都找到,每找到一个就比较一下。
  • 怎么全排列,在递归专题哪里有。但是这样时间复杂度太高了。

解法一:利用哈希表

  • 可以用 两个hash表记录两个字符串中每个字符出现的次数
  • 最后在比较一下两个hash表是否相等就行了。

我们可以用数组模拟hash表。

class Solution {
public:
    bool CheckPermutation(string s1, string s2) 
    {
        //两个hash
        unordered_map<char,int> hash1,hash2;
        if(s1.size()!=s2.size()) return false;

        for(char& c:s1)
            hash1[c]++;
        for(char& c:s2)
            hash2[c]++;
        
//比较两hash表
       for (const auto& [key, val] : hash1) {
        auto it = hash2.find(key);
        if (it == hash2.end() || it->second != val) 
            return false;
    }
    return true;
    }
};
  • find return pair
  • count return value

优化一下:只使用一个hash表。

  • 将s1字符串中字符出现次数放到hash表
  • 然后取出s2字符串中每个字符,对hash表字符对应位置进行--操作。
  • 如果出现<0,说明就不是字符重排。
  • 还有当s1和s2字符串长度不相等也不是字符重排。
class Solution {
public:
    bool CheckPermutation(string s1, string s2) 
    {
        unordered_map<char,int> hash;
        if(s1.size()!=s2.size()) return false;
        for(char& c:s1)
             hash[c]++;
        for(char& c:s2)
        {
            hash[c]--;
            if(hash[c]<0)
                return false;
        }
        return true;
        
    }
};

4.存在重复元素

题目链接:217. 存在重复元素

题目描述:

给你一个整数数组 nums 。如果任一值在数组中出现 至少两次 ,返回 true ;如果数组中每个元素互不相同,返回 false

示例 1:

输入:nums = [1,2,3,1]

输出:true

解释:

元素 1 在下标 0 和 3 出现。

示例 2:

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

输出:false


题解

解法:哈希表

这道题和最开始那道题的解题思路是一样的。

  • 前面那道题固定一个数往前找找有没有出现 target - nums[i] 的数,遍历找太麻烦了,因此来一个hash表快速查找前面有没有出现target - nums[i] 的数。
  • 这道题 固定一个数 往前找 找有没有出现和我一样的数

也是利用哈希表快速查找。

class Solution {
public:
    bool containsDuplicate(vector<int>& nums) 
    {
        //!!!往前找
        unordered_map<int,int> hash;
        for(int i=0;i<nums.size();i++)
        {
            if(hash.count(nums[i]))
                return true;
            hash[nums[i]]=i;
        }
        return false;
        
    }
};

5.存在重复元素 II

题目链接: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

题解

  • 上一道题仅是找到重复元素即可,但是这道题不仅要是重复元素
  • 而且重复元素的下标必须满足 abs(i - j) <= k

解法:哈希

  • 还是固定一个数,往前找,看看有没有出现重复元素,可以用hash快速查找
  • 不过这道题有点不一样的就是,我们还需要知道元素对应的下标
  • 因此可以使用unordered_map<int,int>,记录nums[i] 对应的下标。
  • 这里有一个细节问题,两个相同元素下标之差要小于等于k,unordered_map不允许插入相同的元素,value会覆盖。

但是这 正好满足我们abs(i - j) <= k,因为我们就是想要找距离小的。

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

⭕6.字母异位词分组

题目链接: 49. 字母异位词分组

题目分析:

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

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

示例 1:

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

示例 2:

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

示例 3:

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

题解

  • 字母异位词:原有字符串的字符重新排序得到的新的字符串。
  • 这道题要求将属于同一个字母异位词放在一块。

解法:哈希表

这道题我们要解决的有两个问题:

  • 如何判断两个字符串是字母异位词
  • 如何分组

如何判断两个字符串是字母异位词,我们可以用之前的思路

  • 弄一个hash表统计每个字符出现的次数。不过代码写着有点麻烦。这里我们可以将字符串按照ascll码 排序!
  • 如果是 相同的字母异位词排完序 后是一样的。

如何分组?

分组是将相同的字母异位词添加到属于自己的数组中。因此这里我们可以来一个unordered_map<string,vector>

  • string 排序后的字符串
  • vector<string> 是属于相同字母异位词的数组。

class Solution {
public:
    vector<vector<string>> groupAnagrams(vector<string>& strs) 
    {
        //如何实现hash排序
        unordered_map<string,vector<string>> hash;
        vector<vector<string>> ret;

        for(auto& s:strs)
        {
            string str=s;
            sort(str.begin(),str.end());
    //!!排序后 插入
            hash[str].push_back(s);
        }

        for(auto& [x,y]:hash)
        {
            ret.push_back(y);
        }
        return ret;
    }
};

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

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

相关文章

缓存之美:Guava Cache 相比于 Caffeine 差在哪里?

大家好&#xff0c;我是 方圆。本文将结合 Guava Cache 的源码来分析它的实现原理&#xff0c;并阐述它相比于 Caffeine Cache 在性能上的劣势。为了让大家对 Guava Cache 理解起来更容易&#xff0c;我们还是在开篇介绍它的原理&#xff1a; Guava Cache 通过分段&#xff08;…

小组件适配屏幕主题色

iOS 18 新增Home screen Tint Color&#xff08;色调&#xff09;选择&#xff0c;用户可以通过以下方式自定义主屏幕颜色&#xff0c;并且小组件&#xff0c;APP 图标也会跟随改颜色。 比如说意料之外的小组件&#xff08;不兼容&#xff09; 白色部分内部应该还有其他显示内…

IO学习---->线程

1.创建两个线程&#xff0c;分支线程1拷贝文件的前一部分&#xff0c;分支线程2拷贝文件的后一部分 #include <head.h> sem_t sem; long half_size 0; // 全局变量&#xff0c;供所有线程共享void* product(void *arg) {FILE *src fopen("IO.text", "…

个人记录,Unity资源解压和管理插件

就是经典的两个AssetStudio 和 Ripper 没有什么干货&#xff0c;就是记录一下&#xff0c;内容没有很详细 AssetStudio 说错了&#xff0c;AssetStudio比较出名&#xff08;曾经&#xff09;&#xff0c;但好像堕落了 这个工具有个好处就是分类选择&#xff0c;&#xff08;…

day19-前端Web——Vue3+TS+ElementPlus

目录 1. Vue工程化1.1 介绍1.2 环境准备1.2.1 NodeJS安装双击安装包选择安装目录验证NodeJS环境变量配置npm的全局安装路径 1.3 Vue项目-创建1.4 Vue项目开发流程1.5 API风格1.6 案例 2. TS2.1 概述2.2 快速入门2.3 常用类型2.3.1 基础类型2.3.2 联合类型2.3.3 函数类型2.3.4 对…

隐私保护在 Facebook 用户身份验证中的应用

在这个数字化的时代&#xff0c;个人隐私保护成为了公众关注的焦点。社交媒体巨头 Facebook 作为全球最大的社交平台之一&#xff0c;拥有数十亿用户&#xff0c;其在用户身份验证过程中对隐私保护的重视程度直接影响着用户的安全感和信任度。本文将探讨 Facebook 在用户身份验…

【JavaWeb学习Day23】

Maven高级 分模块设计与开发 分模块设计&#xff1a;将一个大项目分成若干个子模块&#xff0c;方便项目的维护、扩展&#xff0c;也方便模块间的相互引用&#xff0c;资源共享。 策略&#xff1a; 1.策略一&#xff1a;按照功能模块拆分&#xff0c;比如&#xff1a;公共组…

个人记录的一个插件,Unity-RuntimeMonitor

没有什么干货,仅仅是个人的记录 基于GUI做的一个工具:好处就是Monitor必须,Unity天然支持实时的Monitor;唯一不好处,就是默认字体太小了,layout居中,居右也是要自行设计的。 (下面文字是有一点点写错,但意思和功能就很牛逼了;并不是都按2 x shift,而是一个 shift 添…

【NexLM 开源系列】如何封装多个大模型 API 调用

&#x1f31f; 在这系列文章中&#xff0c;我们将一起探索如何搭建一个支持大模型集成项目 NexLM 的开发过程&#xff0c;从 架构设计 到 代码实战&#xff0c;逐步搭建一个支持 多种大模型&#xff08;GPT-4、DeepSeek 等&#xff09; 的 一站式大模型集成与管理平台&#xff…

Git和GitHub基础教学

文章目录 1. 前言2. 历史3. 下载安装Git3.1 下载Git3.2 安装Git3.3 验证安装是否成功 4. 配置Git5. Git基础使用5.1 通过Git Bash使用5.1.1 创建一个新的仓库。5.1.1.1 克隆别人的仓库5.1.1.2 自己创建一个本地仓库 5.1.2 管理存档 5.2 通过Visual Studio Code使用 6. Git完成远…

笔记六:单链表链表介绍与模拟实现

在他一生中&#xff0c;从来没有人能够像你们这样&#xff0c;以他的视角看待这个世界。 ---------《寻找天堂》 目录 文章目录 一、什么是链表&#xff1f; 二、为什么要使用链表&#xff1f; 三、 单链表介绍与使用 3.1 单链表 3.1.1 创建单链表节点 3.1.2 单链表的头插、…

坐落于杭州的电商代运营公司品融电商

坐落于杭州的电商代运营公司品融电商 在中国电商行业蓬勃发展的浪潮中&#xff0c;品融电商&#xff08;PINKROON&#xff09;作为一家扎根杭州的新锐品牌管理公司&#xff0c;凭借其独特的全域增长方法论和实战经验&#xff0c;迅速崛起为行业标杆。自2020年成立以来&#x…

【算法学习之路】8.栈和队列

栈和队列 前言一.简介二.题目12 前言 我会将一些常用的算法以及对应的题单给写完&#xff0c;形成一套完整的算法体系&#xff0c;以及大量的各个难度的题目&#xff0c;目前算法也写了几篇&#xff0c;题单正在更新&#xff0c;其他的也会陆陆续续的更新&#xff0c;希望大家点…

OpenMCU(三):STM32F103 FreeRTOS移植

概述 本文主要描述了STM32F103移植FreeRTOS的简要步骤。移植描述过程中&#xff0c;忽略了Keil软件的部分使用技巧。默认读者熟练使用Keil软件。本文的描述是基于OpenMCU_RTOS这个工程&#xff0c;该工程已经下载放好了移植STM32F103 FreeRTOS的所有文件 OpenMCU_RTOS工程的愿景…

大数据 spark hive 总结

Apache Spark 简介 是一个开源的统一分析引擎&#xff0c;专为大规模数据处理而设计。它提供了高级API&#xff0c;支持Java、Scala、Python和R语言&#xff0c;并且包含了一个优化过的执行引擎&#xff0c;该引擎支持循环计算&#xff08;如机器学习算法&#xff09;和交互式…

小程序开发总结

今年第一次帮别人做小程序。 从开始动手到完成上线&#xff0c;一共耗时两天。AI 让写代码变得简单、高效。 不过&#xff0c;小程序和 Flutter 等大厂开发框架差距实在太大&#xff0c;导致我一开始根本找不到感觉。 第一&#xff0c;IDE 不好用&#xff0c;各种功能杂糅在…

QLoggingCategory类使用

QLoggingCategory类使用 QLoggingCategory的概述 QLoggingCategory是Qt的日志策略类&#xff1b;可以通过声明不同的日志策略对象来输出不同的日志信息。打印信息类型如下&#xff1a;宏 Q_DECLARE_LOGGING_CATEGORY(name) 定义一个返回QLoggingCategory对象函数&#xff0c;…

GPU加速生信分析-宏基因组MAG去污染

Deepurify利用多模态深度语言模型来过滤污染的基因组&#xff0c;从而提高了宏基因组组装基因组&#xff08;MAGs&#xff09;的质量&#xff0c;并且可以利用GPU加速。 宏基因组组装的基因组 &#xff08;MAG&#xff09; 为使用宏基因组测序数据探索微生物暗物质提供了有价值…

数据结构(蓝桥杯常考点)

数据结构 前言&#xff1a;这个是针对于蓝桥杯竞赛常考的数据结构内容&#xff0c;基础算法比如高精度这些会在下期给大家总结 数据结构 竞赛中&#xff0c;时间复杂度不能超过10的7次方&#xff08;1秒&#xff09;到10的8次方&#xff08;2秒&#xff09; 空间限制&#x…

从0到1入门Linux

一、常用命令 ls 列出目录内容 cd切换目录mkdir创建新目录rm删除文件或目录cp复制文件或目录mv移动或重命名文件和目录cat查看文件内容grep在文件中查找指定字符串ps查看当前进程状态top查看内存kill终止进程df -h查看磁盘空间存储情况iotop -o直接查看比较高的磁盘读写程序up…