【算法练习Day5】有效的字母异位词 两个数组的交集快乐数两数之和

news2024/11/16 21:52:34

在这里插入图片描述

​📝个人主页:@Sherry的成长之路
🏠学习社区:Sherry的成长之路(个人社区)
📖专栏链接:练题
🎯长路漫漫浩浩,万事皆有期待

文章目录

  • 有效的字母异位词
  • 两个数组的交集
  • 快乐数
  • 两数之和
  • 总结:

这篇博客主要是针对于哈希表的应用内容。哈希表主要分为三种哈希结构:数组,set和map。哈希表的应用主要在于解决一些,需要查找某个元素是否出现过的问题,需要快速查找类问题,可以优先考虑使用哈希表,它的查找时间为O(1)

通常情况,我们在想要一个数据元素差别小,且元素较少的集合时,优先选择数组,它的特点是比set和map存储空间上简单一些,查找数据较快。而面对一些数据元素相差大,且元素数据值较大,或数值多,可以考虑set,无序时预先考虑unordered_set,它是哈希结构实现的,比红黑树实现的set和muliti_set更快。在需要做数据映射也就是同时存储一对有关联的元素时用map。

有效的字母异位词

242. 有效的字母异位词 - 力扣(LeetCode)
在这里插入图片描述

这是用数组做哈希的经典题目,选用数组作为哈希结构的理由是,该题判断两字符串的相同字母出现频率是否相同,而小写字母一共只有26个,数据少且连续,所以选用数组。

解题思路为:哈希表26个字母的位序依次对应于数组的0-25的下标,且一开始均为0,分别遍历两个字符串,对于第一个字符串的每一个字母减去字符‘a’,所得到的就是它当前在哈希表中对应的位置,该字母每出现一次,将该位置数值自增1。遍历第二个字符串就是自减1,和前面处理是相同的方法,处理完之后,若哈希表内0-25位置的元素值均为0,那么说明此时两字符串的字母出现频率相等,如果不为0则直接返回false。

class Solution {
public:
    bool isAnagram(string s, string t) {
        int record[26]={0};
        for(int i=0;i<s.size();i++)
        {
            record[s[i]-'a']++; // 并不需要记住字符a的ASCII,只要求出一个相对数值就可以了
        }
        for(int j=0;j<t.size();j++)
        {
            record[t[j]-'a']--;
        }
        for(int k=0;k<26;k++)
        {
           if(record[k]!=0)// record数组如果有的元素不为零0,说明字符串s和t 一定是谁多了字符或者谁少了字符。
           {
               return false;
           }
        }
        return true;// record数组所有元素都为零0,说明字符串s和t是字母异位词
    }
};

● 为什么遍历第二个字符串时只考虑碰到计数小于0的情况返回false而不考虑大于0的情况(即s字符串出现了t字符串中没有的字符)?
最开始会判断两个字符串长度是否相等,在两个字符串长度相同的情况下,如果有大于0的情况,一定对应地会出现其他字符小于0。

两个数组的交集

349. 两个数组的交集 - 力扣(LeetCode)
在这里插入图片描述

这道题,也可以用数组来做哈希,且运行效率很高,但是由于此题在未更改题目描述和测试用例时,是没有给定具体的数组长度的,所以用set

具体思路为:定义两个无序set,由于判断两个数组交集返回的数据可以忽略数据的顺序,所以可以使用无序set来提高运行速度。第一个set用来存储第一个数组的所有元素(由于set的特点数组中有重复元素并不会重复存放),存储完之后,遍历第二个数组,用find成员函数来查找该set里有没有和其相同的元素出现,若有则放到第二个set里,同样该set也是可以去重的,最后返回的vector里的元素用第二个set来填充即可。

class Solution {
public:
    vector<int> intersection(vector<int>& nums1, vector<int>& nums2) 
    {
        unordered_set<int> result_set; // 存放结果,之所以用set是为了给结果集去重
        unordered_set<int> nums_set(nums1.begin(), nums1.end());
        for (int num : nums2) 
        {
            // 发现nums2的元素 在nums_set里又出现过
            if (nums_set.find(num) != nums_set.end()) 
            {
                result_set.insert(num);
            }
        }
        return vector<int>(result_set.begin(), result_set.end());
    }
};

当然用数组来做哈希结构也是可以的,这里只提供思路:用数组哈希来存储nums1中每个数出现的频率,再判断该哈希中的数是否在nums2里也出现过,将值传进set内,完成后return。

快乐数

202. 快乐数 - 力扣(LeetCode)
在这里插入图片描述

快乐数是求一个数的每一位求平方和,最后经过这样的若干次变化后是否等于1。

解题思路的要点是用set来存储这些中间变化的平方和值,那么为什么要这样做呢?一个数只分为经过变化后平方和等于1循环,和经过变化后平方和变成某些数字的死循环两种。所以用哈希存储变化中的数值至关重要。由于不知道该数要变换几次,可能次数很多需要存储数据很多,所以采用set来做哈希。

代码如下

class Solution {
public:
    // 取数值各个位上的单数之和
    int getSum(int n) 
    {
        int sum = 0;
        while (n) 
        {
            sum += (n % 10) * (n % 10);
            n /= 10;
        }
        return sum;
    }
    bool isHappy(int n) 
    {
        unordered_set<int> set;
        while(1) 
        {
            int sum = getSum(n);
            if (sum == 1) 
            {
                return true;
            }
            // 如果这个sum曾经出现过,说明已经陷入了无限循环了,立刻return false
            if (set.find(sum) != set.end()) 
            {
                return false;
            } 
            else 
            {
                set.insert(sum);
            }
            n = sum;
        }
    }
};

当判断中取得的平方和在set中已经被取到,则直接返回false。

另一种思路是leetcode官方给出的,双指针方法,很巧妙。

思路是:创建slow和fast两个变量,起初都等于数字n,然后用平方和函数调用一下,slow调用一次,fast调用两次,最终无论是什么结果两指针都会相遇,即指向相同的数字,如果是结果为1结束的,那么1的平方和无论怎么加都是1,如果是一些数的循环,可以把它想象为一个环形链表,在固定的一些数里循环,总有slow和fast相等的情况。出循环时判断一个指针是否等于1就可以了。

class Solution {
public:
    bool isHappy(int n) 
    {
        int slow=n;int fast=n;
        do{
            slow=get(slow);
            fast=get(fast);
            fast=get(fast);
        }
        while(slow!=fast);
        
        return (slow==1);
    }
    int get(int n)
    {
        int sum=0;
        while(n)
        {
            sum+=(n%10)*(n%10);
            n/=10;
        }
        return sum;
    }
};

● sum重复出现,就肯定不是快乐数,为什么呢?
因为只要重复出现一次就说明会无限循环,就像之前链表那个环,假设a1算完等于a2,a2算完等于a3,a3算完等于a1,那么下一次a1算完必定等于a2,再下一次a2算完必定是a3,形成了一个循环,而这个循环中不可能有1,因为1平方的结果永远是1,所以肯定有循环就肯定不是快乐数

两数之和

1. 两数之和 - 力扣(LeetCode)
在这里插入图片描述

这道题基本是每个刷leetcode的第一道题,我最开始做这道题时候,只写出了一个暴力解法,而且还是看题解写的。暴力的思路简单一些就是两个循环,第一个数和剩下的数相加,第二个数和剩下的数相加,以此类推。

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

第二种方法就是使用哈希法中的map存储结构,为什么用map呢?思路是这样的,我们将数组中已经遍历过的数用map存储下来,由于我们需要返回的是数组下标,而并非是哪两个数所组成的,所以我们将map设置为key值为数组里的元素值,而value设置为数组元素对应的下标。

我们每遍历一个数就要在map里面查找,是否有可以与该数相加之后构成target的值,如果有则直接返回,没有将该数加入map,遍历下一个数,直到遍历完全,若还没有返回,则说明找不到目标数返回null。

class Solution {
public:
    vector<int> twoSum(vector<int>& nums, int target) {
        std::unordered_map <int,int> map;
        for(int i = 0; i < nums.size(); i++) {
            // 遍历当前元素,并在map中寻找是否有匹配的key
            auto iter = map.find(target - nums[i]); 
            if(iter != map.end()) {
                return {iter->second, i};
            }
            // 如果没找到匹配对,就把访问过的元素和下标加入到map中
            map.insert(pair<int, int>(nums[i], i)); 
        }
        return {};
    }
};

● 一般说数组作为哈希表 是利用值作为数组下标来达到快速定位 所以查找也能达到O(1)的复杂度 但是适用范围很有限

● 用unordered_map不是不能存储两个相同的key吗,那如果数组里两个出现相同的两个元素都要存储会怎么样呢?
注意它存入的方式,它是在循环的过程中边检验边存的,如果没有对应的数字就存入map,如果有就计数,这样不会遇到重复的

总结:

今天的四道题涉及哈希的思想,之前没有这么练习过,收获不少。接下来,我们继续进行算法练习·。希望我的文章和讲解能对大家的学习提供一些帮助。

当然,本文仍有许多不足之处,欢迎各位小伙伴们随时私信交流、批评指正!我们下期见~

在这里插入图片描述

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

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

相关文章

基于springboot+vue的青年公寓服务平台

博主主页&#xff1a;猫头鹰源码 博主简介&#xff1a;Java领域优质创作者、CSDN博客专家、公司架构师、全网粉丝5万、专注Java技术领域和毕业设计项目实战 主要内容&#xff1a;毕业设计(Javaweb项目|小程序等)、简历模板、学习资料、面试题库、技术咨询 文末联系获取 项目介绍…

K8SYaml文件详解及编写示例

文章目录 一.Yaml文件详解1.Yaml文件格式2.YAML 语法格式 二.Yaml文件编写及相关概念1.查看 api 资源版本标签2.yaml编写案例&#xff08;1&#xff09;相关标签介绍&#xff08;2&#xff09;Deployment类型编写nginx服务&#xff08;3&#xff09;k8s集群中的port介绍&#x…

Unity 内存性能分析器 (Memory Profiler)

一、 安装 安装有两种方式一&#xff1a; add package : com.unity.memoryprofiler方式二&#xff1a; From Packages : Unity Registry 搜索 Memory Profiler 二、 使用 打开&#xff1a;Windows - > Analysis - > Memory Profiler 打开MemoryProfiler界面&#xff0…

深入学习JVM(Java虚拟机)

目录 一.JDK、JRE、JVM的关系 1.1JDK(Java SE Development Kit) 1.2JRE( Java Runtime Environment) 1.3JVM(Java Virtual Machine) 1.4JDK、JRE、JVM的区别与联系 二.Class的生命周期 2.1加载 2.1.1 类加载器 2.1.2类加载机制 2.1.3双亲委派 2.2链接 2.2.1验证 2…

antd-vue 级联选择器默认值不生效解决方案

一、业务场景&#xff1a; 最近在使用Vue框架和antd-vue组件库的时候&#xff0c;发现在做编辑回显时** 级联选择器** 组件的默认值不生效。为了大家后面遇到和我一样的问题&#xff0c;给大家分享一下 二、bug信息&#xff1a; 三、问题原因&#xff1a; 确定不了唯一的值&a…

GEE:根据影像最小值和最大值自适应可视化参数设置

作者:CSDN @ _养乐多_ 本文将介绍根据影像最小值和最大值,自适应的设置影像可视化参数设置。 文章目录 一、核心函数二、代码示例三、代码链接一、核心函数 //计算影像的最小值和最大值,为了可视化 var imageMin = (image.reduceRegion({reduc

使用CRM管理销售渠道的 5 个重要优势

销售渠道的管理是创造积极客户体验的关键因素&#xff0c;对企业来说非常重要。大多数情况下&#xff0c;客户关系管理&#xff08;CRM&#xff09;系统可用于高效的销售渠道管理。那么&#xff0c;利用CRM系统管理销售渠道有哪些优势呢&#xff1f; 1) 有效的线索管理 销售渠…

CSS实现鼠标悬停图片上升显示

文章目录 前言一、实现效果二、实现思路 前言 当我们想在图片上面放置一些文字内容时&#xff0c;发现不管怎么放置&#xff0c;要么就是图片影响到文字的观感&#xff0c;要么就是文字挡住图片的细节&#xff0c;那么怎么可以既看到图片的细节又可以看到对图片的文字描述呢&a…

基于R语言分位数回归丨线性回归假设与分位数函数、线性分位数回归 、贝叶斯分位数回归、超越线性分位数回归等

目录 专题一 线性回归假设与分位数函数讲解 专题二 线性分位数回归 【代码实践】 专题三 贝叶斯分位数回归【代码实践】 专题四 超越线性分位数回归&#xff08;一&#xff09;【代码实践】 专题五 超越线性分位数回归&#xff08;二&#xff09;【代码实践】 更多应用 回…

怎样的外发文件管理办法 能够避免数据外发泄露?

在日常办公中&#xff0c;重要文件保密管理可谓“老生常谈”。但我们往往容易忽视&#xff0c;文件保密管理并非个体所能独立完成&#xff0c;在整个文件运转过程中&#xff0c;存在多名经手人&#xff0c;一人发生疏忽&#xff0c;则整个安全屏障都会被打破。 因此&#xff0c…

膝盖前交叉韧带断裂

等级、治愈&#xff08;几乎不可能&#xff0c;除非康复阶段做得非常到位。手术后可恢复到90%&#xff09;&#xff0c;饮食&#xff0c;康复训练&#xff08;完全康复&#xff09;、心理作用 赔偿&#xff1a;工商&#xff08;分出两份&#xff09;&#xff0c;伤残证&#x…

Spring Boot 如何实现单点登录(SSO)

当今的应用程序越来越多地采用了微服务架构&#xff0c;这就引出了一个重要的问题&#xff1a;如何实现单点登录&#xff08;Single Sign-On&#xff0c;简称SSO&#xff09;来确保用户在多个微服务之间无需重复登录。Spring Boot是一个流行的Java框架&#xff0c;它提供了一些…

旅行季《乡村振兴战略下传统村落文化旅游设计》许少辉八一新著作想象和世界一样宽广

旅行季《乡村振兴战略下传统村落文化旅游设计》许少辉八一新著作想象和世界一样宽广

AB试验(四)基于规范流程的一个案例分析

AB试验&#xff08;四&#xff09;基于规范流程的一个案例分析 确定目标和假设 业务场景&#xff1a;某音乐APP&#xff0c;通过历史数据发现一些便利功能往往有着较高的留存和续订。但是这些便利功能的使用率并不高。调研发现&#xff0c;由于APP崇尚简洁设计&#xff0c;因…

排序算法二 归并排序和快速排序

目录 归并排序 快速排序 1 挖坑法​编辑 2 Hoare法 快排的优化 快排的非递归方法 七大排序算法复杂度及稳定性分析 归并排序 归并排序是建立在归并操作上的一种有效的排序算法,将以有序的子序列合并,得到完全有序的序列,即先使每个子序列有序,在使子序列段间有序.若将两…

李沐_动手学深度学习_19_卷积层

一、遇到的磕磕绊绊&#xff1a; 1.一维数组 和 二维矩阵数据之间的区别&#xff1a; 二、用到的一些代码&#xff1a; #备注&#xff0c;矩阵点乘 就是 A*B import torch from d2l import torch as d2l #这个库是李沐自己写的&#xff0c;我去 from torch import nndef co…

【React】JSX语法

目录 一、前言二、JSX介绍三、JSX原理1、DOM结构示例2、HTML的JSX结构示例3、编译之后的代码 四、为什么使用JSX1、JSX的特点2、JSX的书写规范 五、JSX的使用1、嵌入JS表达式2、条件渲染3、列表渲染①、arr.map() 六、组件1、类组件①、实例化组件 2、函数组件3、组件样式①、行…

无缝转换:将File转化为MultipartFile,轻松应对文件上传

无缝转换&#xff1a;将File转化为MultipartFile&#xff0c;轻松应对文件上传 1、概述2、文件转换2.1、 什么是 MultipartFile2.2、将 File 对象转换为 MultipartFile 对象 3、总结 1、概述 大家好&#xff0c;我是欧阳方超&#xff0c;可以关注我的公众号“欧阳方超”&#…

Linux 搭建 Oracel 10g 环境

Oracle 操作 1. Linux 安装 oracle 10g (1) 登录系统 操作系统: Kylin 3.2 硬盘空间: 8G 以上 数据库版本: oracle 10.2.0 使用 root 用户登录操作系统&#xff0c;若为普通用户使用su命令切换至 root用户。 (2) 准备文件 将数据库安装文件&#xff08;10201_database_…

字节8年经验之谈 —— 10大自动化测试框架总结!

软件行业正迈向自主、快速、高效的未来。为了跟上这个高速前进的生态系统的步伐&#xff0c;必须加快应用程序的交付时间&#xff0c;但不能以牺牲质量为代价。快速实现质量是必要的&#xff0c;因此质量保证得到了很多关注。为了满足卓越的质量和更快的上市时间的需求&#xf…