当前位置: 首页 > web >正文

【LeetCode Solutions】LeetCode 热题 100 题解(1 ~ 5)

CONTENTS

  • 哈希 - LeetCode 1. 两数之和(简单)
  • 哈希 - LeetCode 49. 字母异位词分组(中等)
  • 哈希 - LeetCode 128. 最长连续序列(中等)
  • 双指针 - LeetCode 283. 移动零(简单)
  • 双指针 - LeetCode 11. 盛最多水的容器(中等)

哈希 - LeetCode 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≤1042\le nums.length\le 10^42nums.length104
−109≤nums[i]≤109-10^9\le nums[i]\le 10^9109nums[i]109
−109≤target≤109-10^9\le target\le 10^9109target109


【分析】

维护一个哈希表,记录 nums[1]∼nums[i−1]nums[1]\sim nums[i-1]nums[1]nums[i1],当遍历到 nums[i]nums[i]nums[i] 时,通过哈希表查找是否存在 target−nums[i]target-nums[i]targetnums[i],如果存在说明找到答案。


【代码】

class Solution {
public:vector<int> twoSum(vector<int>& nums, int target) {unordered_map<int, int> st;  // 维护 nums[i] 及其下标for (int i = 0; i < nums.size(); i++) {int x = target - nums[i];if (st.count(x)) return { st[x], i };st[nums[i]] = i;}return {};  // 为了防止编译出问题}
};

哈希 - LeetCode 49. 字母异位词分组(中等)

【题目描述】

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

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

【示例 1】

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

【示例 2】

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

【示例 3】

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

【提示】

1≤strs.length≤1041\le strs.length\le 10^41strs.length104
0≤strs[i].length≤1000\le strs[i].length\le 1000strs[i].length100
strs[i] 仅包含小写字母


【分析】

我们可以将每个字符串按字典序排序,这样如果两个字符串所含的字符相同那么排序后的字符串一定也相同,然后我们再用一个哈希表记录每类字符串即可。


【代码】

class Solution {
public:vector<vector<string>> groupAnagrams(vector<string>& strs) {vector<vector<string>> res;unordered_map<string, int> st;  // 记录每一类的下标int idx = 0;for (auto s: strs) {string t = s;sort(t.begin(), t.end());if (!st[t]) st[t] = ++idx, res.push_back(vector<string>{s});else res[st[t] - 1].push_back(s);  // 注意 vector 中下标从 0 开始}return res;}
};

哈希 - LeetCode 128. 最长连续序列(中等)

【题目描述】

给定一个未排序的整数数组 nums,找出数字连续的最长序列(不要求序列元素在原数组中连续)的长度。

请你设计并实现时间复杂度为 O(n)O(n)O(n) 的算法解决此问题。

【示例 1】

输入:nums = [100,4,200,1,3,2]
输出:4
解释:最长数字连续序列是 [1, 2, 3, 4]。它的长度为 4。

【示例 2】

输入:nums = [0,3,7,2,5,8,4,6,0,1]
输出:9

【示例 3】

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

【提示】

0<=nums.length<=1050 <= nums.length <= 10^50<=nums.length<=105
−109<=nums[i]<=109-10^9 <= nums[i] <= 10^9109<=nums[i]<=109


【分析】

首先用哈希表记录下每个数是否存在,然后按照任意顺序枚举每个数,假如枚举到 xxx,然后我们就不断查看 x+1,x+2,…,yx + 1, x + 2, \dots, yx+1,x+2,,y 是否存在,直到 y+1y + 1y+1 不存在,那么连续序列的长度就为 y−x+1y - x + 1yx+1

但如果每次都这样枚举时间复杂度为 O(n2)O(n^2)O(n2),需要优化,我们只枚举每段连续序列的起点,也就是最小值,如果 x−1x - 1x1 存在说明 xxx 肯定不是最长连续序列的起点,就不用枚举了。

此外还需要进行判重,对于遍历过的序列将其从哈希表中删去,保证每个数只被枚举一次,这样时间复杂度就为 O(n)O(n)O(n)


【代码】

class Solution {
public:int longestConsecutive(vector<int>& nums) {unordered_set<int> st(nums.begin(), nums.end());for (int num: nums) st.insert(num);int res = 0;for (int num: nums)if (st.count(num) && !st.count(num - 1)) {  // num 为连续序列的起点int x = num;while (st.count(x)) st.erase(x++);  // 将 num 为起点的连续序列从哈希表中删去res = max(res, x - num);}return res;}
};

双指针 - LeetCode 283. 移动零(简单)

【题目描述】

给定一个数组 nums,编写一个函数将所有 0 移动到数组的末尾,同时保持非零元素的相对顺序。

请注意,必须在不复制数组的情况下原地对数组进行操作。

【示例 1】

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

【示例 2】

输入: nums = [0]
输出: [0]

【提示】

1<=nums.length<=1041 <= nums.length <= 10^41<=nums.length<=104
−231<=nums[i]<=231−1-2^{31} <= nums[i] <= 2^{31} - 1231<=nums[i]<=2311

进阶:你能尽量减少完成的操作次数吗?


【分析】

非常简单的双指针算法,一个指针 iii 从前往后遍历数组,另一个指针 kkk 从前往后存储非零元素,如果 nums[i]nums[i]nums[i] 非零,就将其移动到位置 kkk 上。


【代码】

class Solution {
public:void moveZeroes(vector<int>& nums) {for (int i = 0, k = 0; i < nums.size(); i++)if (nums[i]) swap(nums[i], nums[k++]);}
};

双指针 - LeetCode 11. 盛最多水的容器(中等)

【题目描述】

给定一个长度为 n 的整数数组 height。有 n 条垂线,第 i 条线的两个端点是 (i, 0)(i, height[i])
找出其中的两条线,使得它们与 x 轴共同构成的容器可以容纳最多的水。
返回容器可以储存的最大水量。
说明:你不能倾斜容器。

【示例 1】

在这里插入图片描述

输入:[1,8,6,2,5,4,8,3,7]
输出:49 
解释:图中垂直线代表输入数组 [1,8,6,2,5,4,8,3,7]。在此情况下,容器能够容纳水(表示为蓝色部分)的最大值为 49。

【示例 2】

输入:height = [1,1]
输出:1

【提示】

n=height.lengthn = height.lengthn=height.length
2≤n≤1052\le n\le 10^52n105
0≤height[i]≤1040\le height[i]\le 10^40height[i]104


【分析】

很巧妙的一道贪心思维题,我们先在最左边和最右边设置两个指针,每次将指针指向的数较小的那个指针往中间靠拢一格,且每次都维护一遍最大值即可。

因为当一个指针往中间移动时,矩形的宽度缩小了,想要面积变大,那肯定需要指针指向的数值(即矩形高度)变大,而矩形的高度的瓶颈在于较短的那一条边,因此移动较小的指针。


【代码】

class Solution {
public:int maxArea(vector<int>& height) {int res = 0;for (int l = 0, r = height.size() - 1; l < r; ) {res = max(res, (r - l) * min(height[l], height[r]));if (height[l] < height[r]) l++;else r--;}return res;}
};
http://www.xdnf.cn/news/16349.html

相关文章:

  • [CSS]让overflow不用按shift可以滚轮水平滚动(纯CSS)
  • 【数据库】AI驱动未来:电科金仓新一代数据库一体机如何重构性能边界?
  • 半相合 - 脐血联合移植
  • Kingbasepostgis 安装实践
  • Go 官方 Elasticsearch 客户端 v9 快速上手与进阶实践*
  • R 语言绘制六种精美热图:转录组数据可视化实践(基于 pheatmap 包)
  • Redis替代方案:腾讯云TDSQL-C内存优化实战,TPS秒上涨
  • 大语言模型生成式人工智能企业应用
  • 水库大坝安全监测的主要内容
  • 微算法科技(NASDAQ:MLGO)采用分布式哈希表优化区块链索引结构,提高区块链检索效率
  • mac下 vscode 运行 c++无法弹出窗口
  • 《C++初阶之STL》【vector容器:详解 + 实现】
  • 智能问答分类系统:基于SVM的用户意图识别
  • Android Paging 分页加载库详解与实践
  • 航段导航计算机 (Segment_Navigator) 设计与实现
  • 重构 MVC:让经典架构完美适配复杂智能系统的后端业务逻辑层(内附框架示例代码)
  • 【MacOS】发展历程
  • HTTP 请求方法有哪些?
  • 《基于电阻抗断层扫描(EIT)驱动的肌肉骨骼模型表征人体手臂动态意图用于人机交互》论文解读
  • 当人机交互迈向新纪元:脑机接口与AR/VR/MR的狂飙之路
  • Spring Cloud Gateway 服务网关
  • 2025年第四届创新杯(原钉钉杯)赛题浅析-助攻快速选题
  • Android Studio 2024 内嵌 Unity 3D 开发示例
  • 【第四章:大模型(LLM)】01.神经网络中的 NLP-(1)RNN、LSTM 和 GRU 的基本原理和应用
  • 全国产化5G-A低空经济基座
  • 【Unity笔记】OpenXR 之VR串流开发笔记:通过RenderTexture实现仅在PC端展示UI,在VR眼镜端隐藏UI
  • 大模型进阶面试题
  • 车载 CAN-Bus 数据记录仪说明书
  • 【C语言进阶】一篇文章教会你文件的读写
  • 【unitrix】 6.16 非负整数类型( TUnsigned )特质(t_unsingned.rs)