爆款云主机2核4G限时秒杀,88元/年起!
查看详情

活动

天翼云最新优惠活动,涵盖免费试用,产品折扣等,助您降本增效!
热门活动
  • 618智算钜惠季 爆款云主机2核4G限时秒杀,88元/年起!
  • 免费体验DeepSeek,上天翼云息壤 NEW 新老用户均可免费体验2500万Tokens,限时两周
  • 云上钜惠 HOT 爆款云主机全场特惠,更有万元锦鲤券等你来领!
  • 算力套餐 HOT 让算力触手可及
  • 天翼云脑AOne NEW 连接、保护、办公,All-in-One!
  • 中小企业应用上云专场 产品组合下单即享折上9折起,助力企业快速上云
  • 息壤高校钜惠活动 NEW 天翼云息壤杯高校AI大赛,数款产品享受线上订购超值特惠
  • 天翼云电脑专场 HOT 移动办公新选择,爆款4核8G畅享1年3.5折起,快来抢购!
  • 天翼云奖励推广计划 加入成为云推官,推荐新用户注册下单得现金奖励
免费活动
  • 免费试用中心 HOT 多款云产品免费试用,快来开启云上之旅
  • 天翼云用户体验官 NEW 您的洞察,重塑科技边界

智算服务

打造统一的产品能力,实现算网调度、训练推理、技术架构、资源管理一体化智算服务
智算云(DeepSeek专区)
科研助手
  • 算力商城
  • 应用商城
  • 开发机
  • 并行计算
算力互联调度平台
  • 应用市场
  • 算力市场
  • 算力调度推荐
一站式智算服务平台
  • 模型广场
  • 体验中心
  • 服务接入
智算一体机
  • 智算一体机
大模型
  • DeepSeek-R1-昇腾版(671B)
  • DeepSeek-R1-英伟达版(671B)
  • DeepSeek-V3-昇腾版(671B)
  • DeepSeek-R1-Distill-Llama-70B
  • DeepSeek-R1-Distill-Qwen-32B
  • Qwen2-72B-Instruct
  • StableDiffusion-V2.1
  • TeleChat-12B

应用商城

天翼云精选行业优秀合作伙伴及千余款商品,提供一站式云上应用服务
进入甄选商城进入云市场创新解决方案
办公协同
  • WPS云文档
  • 安全邮箱
  • EMM手机管家
  • 智能商业平台
财务管理
  • 工资条
  • 税务风控云
企业应用
  • 翼信息化运维服务
  • 翼视频云归档解决方案
工业能源
  • 智慧工厂_生产流程管理解决方案
  • 智慧工地
建站工具
  • SSL证书
  • 新域名服务
网络工具
  • 翼云加速
灾备迁移
  • 云管家2.0
  • 翼备份
资源管理
  • 全栈混合云敏捷版(软件)
  • 全栈混合云敏捷版(一体机)
行业应用
  • 翼电子教室
  • 翼智慧显示一体化解决方案

合作伙伴

天翼云携手合作伙伴,共创云上生态,合作共赢
天翼云生态合作中心
  • 天翼云生态合作中心
天翼云渠道合作伙伴
  • 天翼云代理渠道合作伙伴
天翼云服务合作伙伴
  • 天翼云集成商交付能力认证
天翼云应用合作伙伴
  • 天翼云云市场合作伙伴
  • 天翼云甄选商城合作伙伴
天翼云技术合作伙伴
  • 天翼云OpenAPI中心
  • 天翼云EasyCoding平台
天翼云培训认证
  • 天翼云学堂
  • 天翼云市场商学院
天翼云合作计划
  • 云汇计划
天翼云东升计划
  • 适配中心
  • 东升计划
  • 适配互认证

开发者

开发者相关功能入口汇聚
技术社区
  • 专栏文章
  • 互动问答
  • 技术视频
资源与工具
  • OpenAPI中心
开放能力
  • EasyCoding敏捷开发平台
培训与认证
  • 天翼云学堂
  • 天翼云认证
魔乐社区
  • 魔乐社区

支持与服务

为您提供全方位支持与服务,全流程技术保障,助您轻松上云,安全无忧
文档与工具
  • 文档中心
  • 新手上云
  • 自助服务
  • OpenAPI中心
定价
  • 价格计算器
  • 定价策略
基础服务
  • 售前咨询
  • 在线支持
  • 在线支持
  • 工单服务
  • 建议与反馈
  • 用户体验官
  • 服务保障
  • 客户公告
  • 会员中心
增值服务
  • 红心服务
  • 首保服务
  • 客户支持计划
  • 专家技术服务
  • 备案管家

了解天翼云

天翼云秉承央企使命,致力于成为数字经济主力军,投身科技强国伟大事业,为用户提供安全、普惠云服务
品牌介绍
  • 关于天翼云
  • 智算云
  • 天翼云4.0
  • 新闻资讯
  • 天翼云APP
基础设施
  • 全球基础设施
  • 信任中心
最佳实践
  • 精选案例
  • 超级探访
  • 云杂志
  • 分析师和白皮书
  • 天翼云·创新直播间
市场活动
  • 2025智能云生态大会
  • 2024智算云生态大会
  • 2023云生态大会
  • 2022云生态大会
  • 天翼云中国行
天翼云
  • 活动
  • 智算服务
  • 产品
  • 解决方案
  • 应用商城
  • 合作伙伴
  • 开发者
  • 支持与服务
  • 了解天翼云
      • 文档
      • 控制中心
      • 备案
      • 管理中心

      蓝桥杯算法竞赛系列第九章·巧解哈希题,用这3种数据类型足矣

      首页 知识中心 大数据 文章详情页

      蓝桥杯算法竞赛系列第九章·巧解哈希题,用这3种数据类型足矣

      2025-04-22 09:27:17 阅读次数:4

      key,map,nums,哈希,数组

      哈希基础概念

      常用作哈希表的三种数据结构:

      • 数组
      • set
      • map

      数组的话就不多说了,下面我们来谈谈set和map:

      首先来说说unordered_set和set以及multiset的区别:

      unordered_set底层实现是哈希表,set和multiset的底层是红黑树,而红黑树是一种平衡二叉搜索树,所以key是有序的,但是key不可以修改,因为改动key会导致整棵树的错乱,所以只能删除或者增加。

      具体如下表:

      集合 底层实现 是否有序 数值是否可以重复 能否更改数值 查询效率 增删效率
      set 红黑树 有序 否 否 O(logN) O(logN)
      multiset 红黑树 有序 是 否 O(logN) O(logN)
      unordered_set 哈希表 无序 否 否 O(1) O(1)

      接下来谈谈unordered_map和map以及multimap的区别:

      unordered_map底层实现是哈希表,map和multimap的底层是红黑树,而红黑树是一种平衡二叉搜索树,同理map和multimap的key也是有序的。

      映射 底层实现 是否有序 数值是否可以重复 能否更改数值 查询效率 增删效率
      map 红黑树 key有序 key不可重复 key不可修改 O(logN) O(logN)
      multimap 红黑树 key有序 key可重复 key不可修改 O(logN) O(logN)
      unordered_map 哈希表 key无序 key不可重复 key不可修改 O(1) O(1)

      所以当我们要使用集合来解决哈希问题的时候,优先使用unordered_set,因为它的查询和增删效率都是最优的,如果要求集合是有序的,那么就使用set,如果要求集合不仅有序,还要有重复数值,那么就使用multiset。

      map是一个<key, value>的数据结构,map对key是有限制的,因为其不可以修改,对value是没有限制的,因为key的存储方式是使用红黑树实现的。

      但是哈希法是牺牲空间换取了时间,因为我们要使用额外的数组、set或者map来存放数据,才能实现快速的查找。

      当遇到需要判断一个元素是否在集合中出现过的场景,就应该第一时间想到哈希法。


      哈希相关题目

      下面开始我们刷题之旅,在刷题中探索算法的奥秘~

      · 有效的字母异位词

      题目链接:有效的字母异位词

      解题思路:

      首先我们看到字符串s和t都只包含小写字母,所以我们可以定义一个整数数组用于统计字符串中每个字符的出现次数。

      • 先遍历字符串s统计其中每个字符的出现次数;
      • 紧接着再遍历字符串t用哈希数组中统计的s字符串中字符出现次数减去t中出现次数;
      • 最后再遍历哈希数组,如果数组中每个元素都是0就是有效字母异位词。

      代码详解:

      class Solution {
      public:
          bool isAnagram(string s, string t) 
          {
              // 定义一个数组用来统计每个字符的出现次数
              int hashNums[26] = {0};
      
              // 遍历字符串s统计每个字符的出现次数
              for(int i = 0; i < s.size(); i++)
              {
                  hashNums[s[i] - 'a']++;
              }
      
              // 遍历字符串t用哈希数组中统计的字符出现次数减去t中出现次数
              for(int i = 0; i < t.size(); i++)
              {
                  hashNums[t[i] - 'a']--;
              }
      
              // 最后再遍历哈希数组,如果都为0就符合题意
              for(int i = 0; i < 26; i++)
              {
                  if(hashNums[i] != 0)
                  {
                      return false;
                  }
              }
      
              return true;
          }
      };
      
      // 思路2:
      // 1. 分别统计每个字符串中的字符出现次数
      // 2. 判断两个字符串中所有字符出现次数是否相等
      class Solution {
      public:
          bool isAnagram(string s, string t) 
          {
              vector<int> nums1 = encode(s);
              vector<int> nums2 = encode(t);
      
              // 判断两个字符串中所有字符出现次数是否相等
              for(int i = 0; i < 26; i++)
              {
                  if(nums1[i] != nums2[i])
                  {
                      return false;
                  }
              }
      
              return true;
          }
      
          // 统计字符串中字符的出现次数
          vector<int> encode(string& s)
          {
              vector<int> hashNums(26);
      
              for(int i = 0; i < s.size(); i++)
              {
                  hashNums[s[i] - 'a']++;
              }
      
              return hashNums;
          }
      };
      

      · 赎金信

      题目链接:赎金信

      解题思路:

      本题跟上一道题很类似,只不过可能需要注意一下对于细节的处理,这里就不多说了,请老铁细品。

      代码详解:

      class Solution {
      public:
          bool canConstruct(string ransomNote, string magazine) 
          {
              vector<int> hashNums(26);
      
              // 先统计字符串magazine中字符的出现次数
              for(int i = 0; i < magazine.size(); i++)
              {
                  hashNums[magazine[i] - 'a']++;
              }
      
              // 遍历字符串ransomNote用哈希数组中统计的字符出现次数减去其中出现次数
              for(int i = 0; i < ransomNote.size(); i++)
              {
                  hashNums[ransomNote[i] - 'a']--;
      
                  if(hashNums[ransomNote[i] - 'a'] < 0)
                      return false;
              }
      
              return true;
          }
      };
      

      · 字母异位词分组

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

      解题思路:参考:《la bu la dong》

      本题也是异位词相关,异位词这类问题的关键在于,如何迅速判断两个字符串是异位词,主要考察我们数据编码和 哈希表的使用:

      是否可以找到一种编码方法,使得字母异位词的编码都相同?找到这种编码方式之后,就可以用一个哈希表存储编码相同的所有异位词,得到最终的答案。

      242. 有效的字母异位词 考察了异位词的编码问题,对字符串排序可以是一种编码方案,如果是异位词,排序后就变成一样的了,但是这样时间复杂度略高,且会修改原始数据。更好的编码方案是利用每个字符的出现次数进行编码,也就是下面的解法代码。

      代码详解:

      class Solution {
      public:
          vector<vector<string>> groupAnagrams(vector<string>& strs) 
          {
              // 建立编码到分组的映射
              unordered_map<string, vector<string>> encodeToGroup;
      
              // 将相同编码的字符串放到一个分组中
              for(auto& str : strs)
              {
                  // 对字符串进行编码
                  string code = encode(str);
      
                  // 将相同编码的字符串放到一起
                  encodeToGroup[code].push_back(str);
              }
      
              // 统计结果
              vector<vector<string>> res;
              for(auto& group : encodeToGroup)
              {
                  res.push_back(group.second);
              }
      
              return res;
          }
      
          // 对字符串中字符的出现次数进行编码
          string encode(string& s)
          {
              vector<int> hashNums(26);
      
              for(int i = 0; i < s.size(); i++)
              {
                  hashNums[s[i] - 'a']++;
              }
      
              string code(hashNums.begin(), hashNums.end());
      
              return code;
          }
      };
      

      · 两个数组的交集

      题目链接:两个数组的交集

      解题思路:

      统计两个数组的交集,而且要保证结果中每个元素都是唯一的,所以第一时间想到的是set相关结构。

      然后本题很简单,有不理解的地方直接看代码注释就行啦。

      代码详解:

      class Solution {
      public:
          vector<int> intersection(vector<int>& nums1, vector<int>& nums2) 
          {
              // 将nums1中数字存放到集合中
              unordered_set<int> set;
              for(int num : nums1)
              {
                  set.insert(num);
              }
      
              // 记录结果
              unordered_set<int> res;
      
              // 遍历数组2进行比较,统计结果
              for(int num : nums2)
              {
                  if(set.count(num))
                  {
                      res.insert(num);
                  }
              }
      
              return vector<int>(res.begin(), res.end());
          }
      };
      

      · 快乐数

      题目链接:快乐数

      解题思路:

      题目中说了会 无限循环,那么也就是说求和的过程中,sum会重复出现,这对解题很重要!

      当我们遇到了要快速判断一个元素是否出现集合里的时候,就要考虑哈希法。

      所以这道题目使用哈希法,来判断这个sum是否重复出现,如果重复了就是return false, 否则一直找到sum为1为止。

      代码详解:

      class Solution {
      public:
      
          // 统计数字的每一位的平方和 - 101
          int getNum(int n)
          {
              int sum = 0;
      
              while(n != 0)
              {
                  sum += (n % 10) * (n % 10);
                  n /= 10;
              }
      
              return sum;
          }
      
          bool isHappy(int n) 
          {
              // 将计算结果记录到集合当中
              unordered_set<int> set;
      
              while(n != 1)
              {
                  int num = getNum(n);
      
                  if(set.count(num))
                  {
                      return false;
                  }
                  else
                  {
                      set.insert(num);
                  }
      
                  n = num;
              }
      
              return true;
          }
      };
      

      · 两数之和

      题目链接:两数之和

      解题思路:

      对于一个元素 nums[i],想知道有没有另一个元素 nums[j] 的值为 target - nums[i],这很简单,我们用一个哈希表记录每个元素的值到索引的映射,这样就能快速判断数组中是否有一个值为 target - nums[i] 的元素了。

      简单说,数组其实可以理解为一个「索引 -> 值」的哈希表映射,而我们又建立一个「值 -> 索引」的映射即可完成此题。

      代码详解:

      class Solution {
      public:
          vector<int> twoSum(vector<int>& nums, int target) 
          {
              // 建立值到索引的映射
              unordered_map<int, int> valToIndex;
      
              for(int i = 0; i < nums.size(); i++)
              {
                  // 查哈希表,看是否有能和 nums[i] 凑出 target 的元素
                  int need = target - nums[i];
                  if(valToIndex.count(need))
                  {
                      return {i, valToIndex[need]};
                  }
      
                  // 存入val->index的映射
                  valToIndex[nums[i]] = i;
              }    
      
              return {-1, -1};
          }
      };
      

      · 四数相加 II

      题目链接:四数相加 II

      解题思路:

      1. 首先定义 一个unordered_map,key放a和b两数之和,value 放a和b两数之和出现的次数。
      2. 遍历大A和大B数组,统计两个数组元素之和,和出现的次数,放到map中。
      3. 定义int变量count,用来统计 a+b+c+d = 0 出现的次数。
      4. 在遍历大C和大D数组,找到如果 0-(c+d) 在map中出现过的话,就用count把map中key对应的value也就是出现次数统计出来。
      5. 最后返回统计值 count 就可以了

      代码详解:

      class Solution {
      public:
          int fourSumCount(vector<int>& nums1, vector<int>& nums2, vector<int>& nums3, vector<int>& nums4) {
              // 统计数组12所有元素的和及其出现次数
              unordered_map<int, int> sumToCount;
              for(int i = 0; i < nums1.size(); i++)
              {
                  for(int j = 0; j < nums2.size(); j++)
                  {
                      int sum = nums1[i] + nums2[j];
                      sumToCount[sum]++;
                  }
              }
      
              // 记录符合题意的结果
              int count = 0;
      
              // 遍历数组34寻找符合条件的组合
              for(int i = 0; i < nums3.size(); i++)
              {
                  for(int j = 0; j < nums4.size(); j++)
                  {
                      int sum = nums3[i] + nums4[j];
                      if(sumToCount.count(0 - sum))
                      {
                          count += sumToCount[0-sum];
                      }
                  }
              }
      
              return count;
          }
      };
      

      · 最长连续序列

      题目链接:最长连续序列

      解题思路:

      这道题最直接的想法就是排序,排序之后连续的序列就很容易找到了。但是排序的时间复杂度是 O(NlogN),而题目要求我们时间复杂度为 O(N),所以我们需要另想办法。

      想找连续序列,首先要找到这个连续序列的开头元素,然后递增,看看之后有多少个元素还在 nums 中,即可得到最长连续序列的长度了。

      由此我们可以想到用空间换时间的思路,把数组元素放到哈希集合里面,然后去寻找连续序列的第一个元素,即可在 O(N) 时间找到答案。

      比方说 nums = [8,4,9,1,3,2],我们先找到 1,然后递增,找到了 2, 3, 4,这就是一个长度为 4 的序列。

      代码详情:

      class Solution {
      public:
          int longestConsecutive(vector<int>& nums) 
          {
              // 转化成哈希集合,方便快速判断是否存在某个元素
              unordered_set<int> set;
              for (int num : nums) 
              {
                  set.insert(num);
              }
      				
            	// 记录结果
              int res = 0;
      
              for (int num : set) {
                  if (set.count(num - 1)) {
                      // num 不是连续子序列的第一个,跳过
                      continue;
                  }
                  // num 是连续子序列的第一个,开始继续计算连续子序列的长度
                  int curNum = num;
                  int curLen = 1;
      
                  while (set.count(curNum + 1)) {
                      curNum += 1;
                      curLen += 1;
                  }
                  // 更新最长连续序列的长度
                  res = max(res, curLen);
              }
      
              return res;
          }
      };
      

      · 查找共用字符

      题目链接:查找共用字符

      解题思路:

      • 我们先定义一个哈希数组记录第一个字符串中字符的出现次数
      • 然后再定义一个哈希数组记录其他字符串中字符的出现次数,循环比较取两者较小值,请参考代码注释进行理解
      蓝桥杯算法竞赛系列第九章·巧解哈希题,用这3种数据类型足矣

      代码详情:

      class Solution {
      public:
          vector<string> commonChars(vector<string>& words) 
          {
              // 记录结果
              vector<string> res;
      
              // 统计第一个字符串中字符的出现次数
              int hash[26] = {0};
      
              for(int i = 0; i < words[0].size(); i++)
              {
                  hash[words[0][i] - 'a']++;
              }
      
              // 统计出第一个字符串之外的字符串中字符的出现次数
              int hashOther[26] = {0};
              
              for(int i = 1; i < words.size(); i++)
              {
                  memset(hashOther, 0, sizeof(hashOther)); // 别忘记每次都要清空
      
                  for(int j = 0; j < words[i].size(); j++)
                  {
                      hashOther[words[i][j] - 'a']++;
                  }
      
                  // 和第一个字符串比较,取较小值
                  for(int i = 0; i < 26; i++)
                  {
                      hash[i] = min(hash[i], hashOther[i]);
                  }
              }
      
              // 将哈希数组中元素不为0的转换为字符后插入到结果字符串中
              for(int i = 0; i < 26; i++)
              {
                  // 注意理解为啥是while而不是if
                  while(hash[i] != 0)
                  {
                      string str(1, i + 'a');
                      res.push_back(str);
                      hash[i]--;
                  }
              }
      
              return res;
          }
      };
      

      · 同构字符串

      题目链接:同构字符串

      解题思路:

      字符串没有说都是小写字母之类的,所以用数组不合适了,用map来做映射。

      使用两个map 保存 s[i] 到 t[j] 和 t[j] 到 s[i] 的映射关系,如果发现对应不上,立刻返回 false

      代码详解:

      class Solution {
      public:
          bool isIsomorphic(string s, string t) 
          {
              // 使用两个map来保存从s[i]到t[j]和从t[j]到s[i]的映射关系
              unordered_map<char, char> map1;
              unordered_map<char, char> map2;
      
              for(int i = 0, j = 0; i < s.size(); i++, j++)
              {
                  if(map1.find(s[i]) == map1.end())
                  {
                      // map1保存着从s[i]到t[j]的映射
                      map1[s[i]] = t[j];
                  }
      
                  if(map2.find(t[j]) == map2.end())
                  {
                      // map2保存着从t[j]到s[i]的映射
                      map2[t[j]] = s[i];
                  }
                  
                  // 发现映射 对应不上,立刻返回false
                  if(map1[s[i]] != t[j] || map2[t[j]] != s[i])
                      return false;
              }
      
              return true;
          }
      };
      
      
      // 思路2: 可以仿照单词规律的思路,请题请看下一题
      class Solution {
      public:
          bool isIsomorphic(string s, string t) 
          {
              if(s.size() != t.size())
              {
                  return false;
              }
      
              unordered_map<char,char> map; // 建立字符到字符的映射
              unordered_set<char> hasWord; // 记录有字符映射字符
      
              for(int i = 0; i < s.size(); i++)
              {
                  if(!map.count(s[i]))
                  {
                      if(hasWord.count(t[i]))
                      {
                          return false;
                      }
      
                      map[s[i]] = t[i];
                  }
                  else
                  {
                      if(map[s[i]] != t[i])
                      {
                          return false;
                      }
                  }
      
                  hasWord.insert(t[i]);
              }
      
              return true;
          }
      };
      

      · 单词规律

      题目链接:单词规律

      解题思路:

      利用哈希表,把 pattern 中的每个叠词模式字符在 s 中的对应单词记录下来,就能判断 s 是否匹配 pattern 的模式了。

      具体请看代码详解:

      class Solution {
      public:
          bool wordPattern(string pattern, string s) 
          {
              vector<string> words;
      
              stringstream ss(s);
              string str;
      
              while(getline(ss, str, ' '))
              {
                  words.push_back(str);
              }
      
              // 注意这一条判断语句,别忘记了
              if(pattern.size() != words.size())
              {
                  return false;
              }
      
              // 建立字符到字符串的映射
              unordered_map<char, string> patternToStr;
      
              // 记录已经有字符对应的字符串
              unordered_set<string> hasWord;
      
              for(int i = 0; i < pattern.size(); i++)
              {
                  char c = pattern[i]; 
                  string word = words[i];
      
                  // 该字符还没有建立对应的单词映射
                  if(!patternToStr.count(c))
                  {
                      // 但是与之对应的单词却有自己的字符了
                      if(hasWord.count(word))
                      {
                          return false;
                      }   
      
                      // 建立该字符到单词的映射
                      patternToStr[c] = word;
                  }
                  else // 该字符已经有对应的单词了
                  {   
                      // 如若该字符所对应的单词与实际不符
                      if(patternToStr[c] != word)
                      {
                          return false;
                      }
                  }
      
                  hasWord.insert(word);
              }
      
              return true;
          }
      };
      

      · 字节跳动面试:缺失的第一个正数

      题目链接:缺失的第一个正数

      解题思路:

      理想情况下每个元素都在自己的位置上,如下:

      蓝桥杯算法竞赛系列第九章·巧解哈希题,用这3种数据类型足矣

      现在将元素对应位置打乱,并且缺失了一个元素,如下:

      蓝桥杯算法竞赛系列第九章·巧解哈希题,用这3种数据类型足矣

      题目就是要求我们找到这个缺失的元素,我们该如何找呢?

      注意:首先我们需要遍历数组中的每一个元素,比如nums[i],我们将以nums[i]为下标的元素中的值置换为其绝对值的相反数,也就是做标记。(做这道题目的时候一定要画图来理解)

      像下面这样:

      nums[1] = 3,将3这个位置做标记:

      蓝桥杯算法竞赛系列第九章·巧解哈希题,用这3种数据类型足矣

      nums[2] = 1,将1这个位置做标记:

      蓝桥杯算法竞赛系列第九章·巧解哈希题,用这3种数据类型足矣

      nums[3] = 5,将5这个位置做标记:

      蓝桥杯算法竞赛系列第九章·巧解哈希题,用这3种数据类型足矣

      nums[4] = 1,将1这个位置做标记:

      蓝桥杯算法竞赛系列第九章·巧解哈希题,用这3种数据类型足矣

      nums[5] = 6,将6这个位置做标记:

      蓝桥杯算法竞赛系列第九章·巧解哈希题,用这3种数据类型足矣

      nums[6] = 4,将4这个位置做标记:

      蓝桥杯算法竞赛系列第九章·巧解哈希题,用这3种数据类型足矣

      现在我们看到,只有2这个位置是没有做标记的,所以2就是所求。

      温馨提示:可能比较难以理解,一定要自己动手画图去理解。

      所以想到这一步的话,那么这道题目前的难点就是怎么做标记了,OK接着看:

      首先,我们遍历数组将数组中值为非正整数的元素全部忽略,什么意思呢就是将其值全部映射到n之外,所以我们遍历一遍数组,将值为负数和值为0的元素全部赋值为n+1;

      接下来就是,遍历数组中的每一个元素nums[i],先判断其值是否大于n,若大于,则说明其之前是非正整数,忽略它,继续向后遍历;反之,则将以nums[i]为下标的值置换为其绝对值的相反数(因为可能有两个相同的元素,注意画图理解)。

      最后再遍历数组中的每一个元素,值不为负数,其下标就是题目所求。

      题目详解:

      class Solution {
      public:
          int firstMissingPositive(vector<int>& nums) 
          {   
              int n = nums.size();
      
              // 将数组中的非正整数元素忽略,即是映射到n之外
              for(int i = 0; i < n; i++)
              {
                  if(nums[i] <= 0)
                  {
                      nums[i] = n + 1;
                  }
              }
      
              // 遍历数组中的元素,将以元素的值为对应的下标里的元素取其绝对值的相反数,也就是做标记
              for(int i = 0; i < n; i++)
              {
                  if(abs(nums[i]) <= n) // 说明是正整数
                  {
                      nums[abs(nums[i]) - 1] = -abs(nums[abs(nums[i]) - 1]);
                  }
              }
      
              // 遍历数组中的元素,元素值不为负数的其下标就是答案
              for(int i = 0; i < n; i++)
              {
                  if(nums[i] > 0)
                  {
                      return i + 1;
                  }
              }
      
              return n + 1;
          }
      };
      
      版权声明:本文内容来自第三方投稿或授权转载,原文地址:https://bit-runout.blog.csdn.net/article/details/134267020,作者:安然无虞,版权归原作者所有。本网站转在其作品的目的在于传递更多信息,不拥有版权,亦不承担相应法律责任。如因作品内容、版权等问题需要同本网站联系,请发邮件至ctyunbbs@chinatelecom.cn沟通。

      上一篇:传统CV算法——边缘检测算法综述

      下一篇:【Linux】使用 iptables 验证访问HDFS 所使用到的端口

      相关文章

      2025-05-19 09:04:14

      复杂度的OJ练习

      复杂度的OJ练习

      2025-05-19 09:04:14
      代码 , 复杂度 , 思路 , 数组 , 算法
      2025-05-16 09:15:24

      Redis Hash哈希

      Redis Hash哈希

      2025-05-16 09:15:24
      field , hash , Redis , value , 哈希
      2025-05-16 09:15:24

      如何将一串数字用函数的方法倒过来(C语言)

      如何将一串数字用函数的方法倒过来(C语言)

      2025-05-16 09:15:24
      函数 , 数字 , 数组
      2025-05-16 09:15:24

      jQuery遍历对象、数组、集合

      jQuery遍历对象、数组、集合

      2025-05-16 09:15:24
      jQuery , 对象 , 数组 , 遍历 , 集合
      2025-05-16 09:15:17

      递归,搜索,回溯算法(3)之穷举,暴搜,深搜,回溯,剪枝

      递归,搜索,回溯算法(3)之穷举,暴搜,深搜,回溯,剪枝

      2025-05-16 09:15:17
      回溯 , 子集 , 数组 , 算法 , 递归
      2025-05-14 10:33:31

      计算机小白的成长历程——数组(1)

      计算机小白的成长历程——数组(1)

      2025-05-14 10:33:31
      strlen , 个数 , 元素 , 内存 , 十六进制 , 地址 , 数组
      2025-05-14 10:33:31

      计算机小白的成长历程——习题演练(函数篇)

      计算机小白的成长历程——习题演练(函数篇)

      2025-05-14 10:33:31
      函数 , 字符串 , 数组 , 知识点 , 编写 , 迭代 , 递归
      2025-05-14 10:33:25

      超级好用的C++实用库之sha256算法

      SHA-256,英文全称为Secure Hash Algorithm 256-bit,是一种广泛使用的密码散列函数,属于SHA-2家族。

      2025-05-14 10:33:25
      CHP , 参数 , 哈希 , 算法 , 输入
      2025-05-14 10:03:05

      30天拿下Rust之HashMap

      HashMap,被称为哈希表或散列表,是一种可以存储键值对的数据结构。它使用哈希函数将键映射到存储位置,以便可以快速检索和更新元素。

      2025-05-14 10:03:05
      HashMap , 使用 , 哈希 , 引用 , 方法 , 遍历 , 键值
      2025-05-14 10:02:48

      typescript 将数组清空

      在TypeScript或JavaScript开发中,数组是用于存储和管理一组数据的基础数据结构。当需要清空一个数组时,有多种方法可以实现,而选择合适的方法不仅影响代码的可读性,还会对性能产生一定的影响。不同场景下,选择适合的清空数组的方法至关重要。

      2025-05-14 10:02:48
      length , pop , 引用 , 数组 , 方法
      查看更多
      推荐标签

      作者介绍

      天翼云小翼
      天翼云用户

      文章

      33561

      阅读量

      5251146

      查看更多

      最新文章

      递归,搜索,回溯算法(3)之穷举,暴搜,深搜,回溯,剪枝

      2025-05-16 09:15:17

      DS初阶:二叉树的顺序结构及堆的实现

      2025-05-08 09:04:49

      【30天玩转python】数据分析与可视化

      2025-05-06 09:19:30

      java.lang.IllegalStateException: Duplicate key异常解决

      2025-04-11 07:11:40

      文心一言 VS 讯飞星火 VS chatgpt (22)-- 算法导论4.2 2题

      2025-04-01 10:29:12

      打砖块。有一个 m x n 的二元网格,其中 1 表示砖块,0 表示空白。

      2025-04-01 10:28:48

      查看更多

      热门文章

      前端项目实战66-数组数据处理详解

      2023-05-12 06:47:16

      20.6.4算法心得(数组运用)

      2023-03-21 10:39:47

      字节输入流读数据 使用字节数组

      2023-03-29 09:42:23

      hadoop:MapReduce之 shuffle过程详解

      2023-06-15 06:24:14

      把一个数组(列表)中的数据逆向反转,python

      2023-04-13 09:31:18

      算法问题-五行缺数

      2022-12-26 09:32:17

      查看更多

      热门标签

      算法 leetcode python 数据 java 数组 节点 大数据 i++ 链表 golang c++ 排序 django 数据类型
      查看更多

      相关产品

      弹性云主机

      随时自助获取、弹性伸缩的云服务器资源

      天翼云电脑(公众版)

      便捷、安全、高效的云电脑服务

      对象存储

      高品质、低成本的云上存储服务

      云硬盘

      为云上计算资源提供持久性块存储

      查看更多

      随机文章

      在一个数组中,任何一个前面的数a,和任何一个后面的数b,如果(a,b)是降序的,就称为逆序对。返回逆序对个数。

      社交网络中的最优邀请策略探究。本文以小红准备开宴会为例,提出一种基于贪心算法和二分查找的解决方案,帮助读者在保证愉悦值不低于k的前提下,最小化宴会的阶层差距。

      给定数组hard和money,长度都为N;hard[i]表示i号的难度, money[i]表示i号工作的收入;

      给定一个正数数组arr长度为n、正数x、正数y。

      【算法】前缀和——前缀和

      文心一言 VS chatgpt (6)-- 算法导论2.3 1~2题

      • 7*24小时售后
      • 无忧退款
      • 免费备案
      • 专家服务
      售前咨询热线
      400-810-9889转1
      关注天翼云
      • 旗舰店
      • 天翼云APP
      • 天翼云微信公众号
      服务与支持
      • 备案中心
      • 售前咨询
      • 智能客服
      • 自助服务
      • 工单管理
      • 客户公告
      • 涉诈举报
      账户管理
      • 管理中心
      • 订单管理
      • 余额管理
      • 发票管理
      • 充值汇款
      • 续费管理
      快速入口
      • 天翼云旗舰店
      • 文档中心
      • 最新活动
      • 免费试用
      • 信任中心
      • 天翼云学堂
      云网生态
      • 甄选商城
      • 渠道合作
      • 云市场合作
      了解天翼云
      • 关于天翼云
      • 天翼云APP
      • 服务案例
      • 新闻资讯
      • 联系我们
      热门产品
      • 云电脑
      • 弹性云主机
      • 云电脑政企版
      • 天翼云手机
      • 云数据库
      • 对象存储
      • 云硬盘
      • Web应用防火墙
      • 服务器安全卫士
      • CDN加速
      热门推荐
      • 云服务备份
      • 边缘安全加速平台
      • 全站加速
      • 安全加速
      • 云服务器
      • 云主机
      • 智能边缘云
      • 应用编排服务
      • 微服务引擎
      • 共享流量包
      更多推荐
      • web应用防火墙
      • 密钥管理
      • 等保咨询
      • 安全专区
      • 应用运维管理
      • 云日志服务
      • 文档数据库服务
      • 云搜索服务
      • 数据湖探索
      • 数据仓库服务
      友情链接
      • 中国电信集团
      • 189邮箱
      • 天翼企业云盘
      • 天翼云盘
      ©2025 天翼云科技有限公司版权所有 增值电信业务经营许可证A2.B1.B2-20090001
      公司地址:北京市东城区青龙胡同甲1号、3号2幢2层205-32室
      • 用户协议
      • 隐私政策
      • 个人信息保护
      • 法律声明
      备案 京公网安备11010802043424号 京ICP备 2021034386号