算法刷题Day6 | 242.有效的字母异位词、349. 两个数组的交集、202. 快乐数、1. 两数之和

目录

  • 0 哈希表 哈希函数
  • 1 有效的字母异位词
    • 1.1 string的回顾
    • 1.2 我的代码
  • 2 两个数组的交集
    • 2.1 unordered_set 介绍
    • 2.2 我的解题(set)
  • 3 快乐数
    • 3.1 我的解题(set)
  • 4 两数之和
    • 4.1 暴力求解
    • 4.2 哈希表(map)

请添加图片描述

  • 🙋‍♂️ 作者:海码007
  • 📜 专栏:算法专栏
  • 💥 标题:算法刷题Day6 | 242.有效的字母异位词、349. 两个数组的交集、202. 快乐数、1. 两数之和
  • ❣️ 寄语:书到用时方恨少,事非经过不知难!

0 哈希表 哈希函数

  • 🎈 文档讲解:代码随想录
  • 🎈 视频讲解:b站视频

下面是我对这两个概念的一些理解:

在这里插入图片描述

想使用哈希方法解决问题的时候(一般遇到判断某个元素是否在某个集合中出现的时候就用哈希表),一般常用如下三种数据结构

  • 数组
  • set(集合)
  • map(映射)

1 有效的字母异位词

  • 🎈 文档讲解:代码随想录
  • 🎈 视频讲解:
  • 🎈 做题状态:顺利写出来

1.1 string的回顾

不足:string字符串的遍历有点忘记了,下面总结一下遍历string字符串的几种方式:

  • 使用 operator [] 进行遍历。 string s; s[0]
  • 使用迭代器遍历,因为string也是属于stl中的容器,所以也可以使用迭代器。 string::iterator sit = s.begin()
  • 使用 auto subS :s 的方式进行遍历
  1. operator [ ]
int StrToInt1(string s)
{int value = 0;for (size_t i = 0; i < s.size(); i++){value *= 10;value += s[i] - '0';}return value;
}
  1. 迭代器
int StrToInt2(string s)
{int value = 0;string::iterator sit = s.begin();while (sit != s.end()){value *= 10;value += *sit - '0';sit++;}return value;
}
  1. 新式for循环 auto
int StrToInt3(string s)
{int value = 0;for (auto e : s){value *= 10;value += e - '0';}return value;
}

1.2 我的代码

思路:首先将 单个字符 和 ASCII码 联系起来,这样利用ASCII码作为数组的下标索引,然后保存对应的 字符数据。
可以发现这就是哈希的思想,计算机不能直接对 字符 进行索引,但是我们可以将字符转换成 整数index 这样保存到数组中就可以直接索引到元素。

class Solution {
public:bool isAnagram(string s, string t) {// a的ASCII码是97,z的ASCII码是122// 新建一个数组,大小为(122-97+1)int arr[26] = {0};// 遍历字符串s,数组对应下标元素加一for (auto subS : s){int index = subS - 97;arr[index]++;}// 遍历字符串t,数组对应下标元素减一for (auto subT : t){int index = subT - 97;arr[index]--;}// 遍历 arr 数组如果元素全部为0,则返回truefor (int i = 0; i < 26; i++){if (arr[i] != 0){return false;}}return true;}
};

2 两个数组的交集

  • 🎈 文档讲解:
  • 🎈 视频讲解:https://www.bilibili.com/video/BV1ba411S7wu/?vd_source=d499e7f3a8e68e2b173b1c6f068b2147
  • 🎈 做题状态:题目意思一开始理解错了,原来只要找到有相同元素的就行。我理解成要找出一个子序列相交

2.1 unordered_set 介绍

当然!让我们谈谈 C++ 中的 std::unordered_set。这是一个有趣的容器,它提供了快速的搜索、插入和删除唯一对象的功能。以下是关键要点:

  1. std::unordered_set 是什么?

    • std::unordered_set 是 C++ 中的一种 关联容器
    • 它保存了一组指定类型(我们称之为 Key)的 唯一对象
    • std::set 不同,std::unordered_set 中的元素 不按特定顺序排序
    • 内部根据它们的哈希值将元素组织到 中。
  2. 复杂度:

    • 搜索、插入和删除操作的 平均时间复杂度 都是常数时间。
    • 实际性能取决于哈希函数的质量和元素数量。
  3. 成员函数:

    • begin()end()empty()size()insert()erase()clear() 等等。
    • 你还可以使用 emplace() 来高效地插入元素。
  4. 哈希:

    • 哈希函数决定了元素属于哪个桶。
    • 如果需要,你可以自定义哈希函数。
  5. 示例用法:

    #include <iostream>
    #include <unordered_set>int main() {std::unordered_set<int> mySet;mySet.insert(42);mySet.insert(17);mySet.insert(99);for (const auto& value : mySet) {std::cout << value << " ";}// 输出:99 42 17return 0;
    }
    
  6. 记住:

    • std::unordered_set 中的元素是 唯一的(没有重复项)。
    • 当你需要快速查找且不关心顺序时,可以使用它。

想要了解更多细节,请查阅 C++ 参考文档 或者探索一些实际示例! 🚀🔍

2.2 我的解题(set)

  1. 使用数组(由于题目中,数组总数量不超过1000,所以可以直接用数组作为哈希表,这样可以提高效率)
    vector<int> intersection(vector<int>& nums1, vector<int>& nums2) {unordered_set<int> result;// 初始化哈希set,这样重复的数字直接去除了int hashMap[1001] = {0};for (int i = 0; i < nums1.size(); i++){// 将 nums1 数组映射到 hashMap 中hashMap[nums1[i]] = 1;}for (int i = 0; i < nums2.size(); i++){// 将 nums1 数组映射到 hashMap 中if (hashMap[nums2[i]] == 1){result.insert(nums2[i]);}}vector<int> v_result(result.begin(), result.end());return v_result;}
  1. 使用 unordered_set 制作哈希表,当哈希表元素总量不确定时,用 unordered_set 容器比较好
class Solution {
public:vector<int> intersection(vector<int>& nums1, vector<int>& nums2) {// 初始化哈希set,这样重复的数字直接去除了unordered_set<int> hashMap(nums1.begin(), nums1.end());unordered_set<int> result;for (int i = 0; i < nums2.size(); i++){auto search = hashMap.find(nums2[i]);if (search != hashMap.end()){// 如果找到一个元素有交集,则插入result中result.insert(nums2[i]);}}vector<int> v_result(result.begin(), result.end());return v_result;}
};

3 快乐数

  • 🎈 文档讲解:https://programmercarl.com/0202.%E5%BF%AB%E4%B9%90%E6%95%B0.html#%E6%80%9D%E8%B7%AF
  • 🎈 视频讲解:
  • 🎈 做题状态:对怎么样进去数的拆分求和都忘记了

3.1 我的解题(set)

注意点: hashMap.find(sum) != hashMap.end() 表示找到相同的元素。 有点反逻辑明明是相同,判断条件里确实 != 。

让我们来解释一下:

  1. 首先,让我们看一下 hashMap.find(sum) 的含义。在 C++ 中,std::unordered_mapfind 函数用于查找给定键(sum)是否存在于哈希表中。如果找到了相应的键,则返回指向该键的迭代器;否则,返回指向哈希表末尾的迭代器(即 hashMap.end())。

  2. 现在,我们来看看条件 hashMap.find(sum) != hashMap.end()

    • 如果 hashMap.find(sum) 返回的迭代器不等于 hashMap.end(),那么表示找到了相同的元素。
    • 如果 hashMap.find(sum) 返回的迭代器等于 hashMap.end(),那么表示没有找到相同的元素。

因此,虽然看起来有点反直觉,但这个条件确实是用于判断是否找到了相同的元素。🙂

class Solution {
public:bool isHappy(int n) {// 将求出来的和放入哈希表中// 当放入时已经有重复出现的数的话,则直接返回false,否则知道sum等于1返回trueunordered_set<int> hashMap;int sum = 0;while(true){// 1. 对n进行拆分求和sum = 0;while(n != 0){cout << "n = " << n << endl;sum += (n%10) * (n%10);n /= 10;}cout << "sum = " << sum << endl;// 当sum为1的时候,直接返回trueif (sum == 1) return true;// 判断是否已经是第二次出现,如果不是,则将sum存入哈希表中auto search = hashMap.find(sum);if ( search != hashMap.end() ){// 如果已经是第二次出现,则直接返回falsereturn false;}hashMap.insert(sum);// 将n重新赋值n = sum;}}
};

4 两数之和

  • 🎈 文档讲解:
  • 🎈 视频讲解:
  • 🎈 做题状态:之前已经用暴力求解做出来了,但是哈希的方式没想出来

4.1 暴力求解

class Solution {
public:std::vector<int> twoSum(std::vector<int>& nums, int target) {std::vector<int> result;int sum;for (int i = 0; i < nums.size() - 1; i++) {for (int j = i + 1; j < nums.size(); j++) { // 修改此处的循环条件sum = nums[i] + nums[j];if (sum == target) {result.push_back(i);result.push_back(j);return result;}}}// 如果没有找到符合条件的索引,可以返回一个空的 vectorreturn result;}
};

4.2 哈希表(map)

本题的重要三点内容:

  • 为什么想到用哈希表
  • 为什么使用map作为哈希表,而不是set
  • 本题map是保存了什么数据,什么是键,什么是值

使用 unordered_set 作为哈希表时,存在一个问题,就是找到了那两个元素之后,只能知道元素的值和其中一个元素的下标,另一个元素的下标未知。因为 unordered_set 只保存了键。缺陷代码如下:

class Solution {
public:std::vector<int> twoSum(std::vector<int>& nums, int target) {unordered_set<int> hashMap(nums.begin(), nums.end());for (int i = 0; i < nums.size(); i++){if (hashMap.find(target - nums[i]) != hashMap.end()){cout << "找到了" << endl;// 无法确定另一个元素的索引值std::vector<int> v_result = {i, target - nums[i]};return v_result;}}return nums;}
};

所以需要使用 unordered_map 容器key用来存放元素值,value用来存放索引。

我的思路:先遍历将 vector 的元素值和索引存入 map 中,然后遍历 vector 判断 target - nums[i] 是否在 map 容器中如果存在,则返回。

代码随想录:直接遍历 vector 判断 target - nums[i] 是否在 map 容器中如果存在则返回,如果不存在则插入当前的值。

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++){auto iter = map.find(target - nums[i]);if (iter != map.end()){return {iter->second, i};}// 将 nums 中的元素值和索引插入 map 中map.insert(pair<int, int> (nums[i], i));}return {};}
};

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

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

相关文章

vue-cli自定义创建项目-eslint依赖冲突解决方式

创建项目步骤 概览&#xff1a; 在安装 npm安装时会报错 npm ERR! code ERESOLVE npm ERR! ERESOLVE could not resolve npm ERR! npm ERR! While resolving: vue/eslint-config-standard6.1.0 npm ERR! Found: eslint-plugin-vue8.7.1 npm ERR! node_modules/eslint-plugin…

公网ip和局域网ip

什么是公网IP&#xff1f; 公网&#xff0c;俗称外网&#xff0c;又被叫做互联网&#xff0c;是连接不同地区局域网或者城域网计算机的通信的远程网络。通常可以跨接很大的物理范围&#xff0c;连接多个地区、城市和国家提供远距离通信&#xff0c;形成全球性的互联网络。因此…

开源好用的所见即所得(WYSIWYG)编辑器:Editor.js

文章目录 特点基于区块干净的数据 界面与交互插件标题和文本图片列表Todo表格 使用安装创建编辑器实例配置工具本地化自定义样式 今天介绍一个开源好用的Web所见即所得(WYSIWYG)编辑器&#xff1a; Editor.js Editor.js 是一个基于 Web 的所见即所得富文本编辑器&#xff0c;它…

爬虫怎么使用代理IP通过HTML和CSS采集数据?

使用爬虫采集数据时&#xff0c;有时为了隐藏真实IP地址或规避某些网站的限制&#xff0c;我们需要使用代理IP。同时&#xff0c;通过HTML和CSS选择器&#xff0c;我们可以定位并提取页面中的特定数据。以下是一个基本的步骤说明&#xff0c;以Python的requests和BeautifulSoup…

nodejs版本过高导致vue-cli项目无法正常运行解决方案

95% emitting CompressionPlugin ERROR Error: error:0308010C:digital envelope routines::unsupported 方法一&#xff1a;在使用 npm run dev之前使用 set NODE_OPTIONS--openssl-legacy-provider Error: error:0308010C:digital envelope routines::unsupported 解决方法…

3.11_C++_day1_作业

作业要求&#xff1a; 程序代码&#xff1a; #include <iostream> #include <string.h>using namespace std;int main() {int a0,b0,c0,d0,e0;//分别记录字符串中的大写&#xff0c;小写&#xff0c;数字&#xff0c;空格&#xff0c;其他字符个数string str;cha…

C++作业day1

2> 试编程 提示并输入一个字符串&#xff0c;统计该字符中大写、小写字母个数、数字个数、空格个数以及其他字符个数 要求使用C风格字符串完成 #include <iostream> #include <string.h>using namespace std;int main() {string str;cout << "请输…

C++学习笔记:红黑树

红黑树 什么是红黑树红黑树的规则红黑树节点的定义红黑树的插入空树插入非空插入条件判断新插入的节点 cur 不为 root 且 parent->_col 为红就需要调整父节点为左 grandf->left parent当uncle节点为红色时,只需要进行颜色调整,即可当uncle为空 或 者存在但是为黑parent …

案例分析篇07:数据库设计相关28个考点(23~28)(2024年软考高级系统架构设计师冲刺知识点总结系列文章)

专栏系列文章推荐: 2024高级系统架构设计师备考资料(高频考点&真题&经验)https://blog.csdn.net/seeker1994/category_12601310.html 【历年案例分析真题考点汇总】与【专栏文章案例分析高频考点目录】(2024年软考高级系统架构设计师冲刺知识点总结-案例分析篇-…

服务器主机云主机在日常维护需要注意的几个点

服务器的日常维护对于确保服务器的稳定运行和安全性非常重要&#xff0c;以下是一些常见的服务器日常维护方面&#xff1a; 定期更新操作系统和软件&#xff1a;确保服务器的操作系统、应用程序以及安全补丁都是最新的&#xff0c;以填补已知的安全漏洞和提高系统性能。监视服务…

【网络安全】-数字证书

数字证书 数字证书是互联网通讯中用于标志通讯各方身份信息的一串数字或数据&#xff0c;它为网络应用提供了一种验证通信实体身份的方式。具体来说&#xff0c;数字证书是由权威的证书授权&#xff08;CA&#xff09;中心签发的&#xff0c;包含公开密钥拥有者信息以及公开密…

硬件工程师面试题梳理-百度硬件面试题

硬件工程师基本职责 在公司里面&#xff0c;硬件工程师的主要职责包括设计、开发和测试硬件系统&#xff0c;以满足产品需求和性能要求。他们负责确保硬件系统的可靠性、稳定性和可维护性&#xff0c;并与软件工程师和其他团队成员合作&#xff0c;以确保硬件和软件的协同工作…

有名信号量、网络协议模型、UDP编程发送端

我要成为嵌入式高手之3月5日Linux高编第十五天&#xff01;&#xff01; ______________________________________________________ 学习笔记 有名信号量 1、创建semget #include <sys/types.h> #include <sys/ipc.h> #include <sys/sem.h> int semget(…

伊芙丽签约实在智能,实在Agent数字员工助力品牌效能飙升

近日&#xff0c;国内知名时尚女装品牌伊芙丽与实在智能达成合作&#xff0c;引入业内领先的平台级自动化产品实在Agent数字员工——取数宝&#xff0c;自动获取天猫、淘宝、抖音等线上平台营销数据&#xff0c;开启全域化营销的“提效之旅”。 实在Agent智能体 伊芙丽集团成立…

java agent技术的注入利用与避坑点

什么是Java agent技术&#xff1f; Java代理&#xff08;Java agent&#xff09;是一种Java技术&#xff0c;它允许开发人员在运行时以某种方式修改或增强Java应用程序的行为。Java代理通过在Java虚拟机&#xff08;JVM&#xff09;启动时以"代理"&#xff08;agent…

分发平台如何支持热更

随着移动应用程序和游戏的迅猛发展&#xff0c;用户对于获得最新功能和修复bug的期望也越来越高。为了满足这一需求&#xff0c;现代的分发平台越来越注重在应用程序或游戏发布后能够支持热更新的功能。热更新是指通过网络直接获取更新并应用到用户设备上&#xff0c;而无需重新…

ubuntu 23开机界面美化教程

效果 方法 GRUB开机界面美化 从上述网站中&#xff0c;查找GRUB Themes分类&#xff0c;并下载GRUB主题包&#xff08;tar.gz格式&#xff09;&#xff0c;如CyberSynchro.tar.gz&#xff1b; 解压下载得到的压缩包&#xff0c;得到CyberSynchro&#xff1b; 将CyberSynchro…

leetcode 热题 100_轮转数组

题解一&#xff1a; 新数组存储&#xff1a;另外用一个数组存储移动后的结果&#xff0c;再复制回原数组。 class Solution {public void rotate(int[] nums, int k) {int[] result new int[nums.length];for (int i 0; i < nums.length; i) {result[(i k) % nums.lengt…

Python自动化测试之Python简介及环境安装配置

经过持续的"内卷"&#xff0c;编程变成测试工程师不可或缺的一项能力&#xff0c;掌握了一门编程语言,使你在面试过程中更有竞争力&#xff0c;是升值加薪的利器。 一、Python发展史 Python 是由 Guido van Rossum 在八十年代末和九十年代初&#xff0c;在荷兰国家数…

springboot同时接收json数据和 MultipartFile

首先测试接口发送方式。。。。。注意发送结构&#xff01; 后端接收RequestPart SaCheckPermission("system:records:add")Log(title "【用药纪录】", businessType BusinessType.INSERT)RepeatSubmit()PostMapping()public R<Void> add( RequestP…