【面试HOT200】数组篇

系列综述:
💞目的:本系列是个人整理为了秋招面试coding部分的,整理期间苛求每个算法题目,平衡可读性代码性能(leetcode运行复杂度均打败80%以上)。
🥰来源:材料主要源于【CodeTopHot200】进行的,每个知识点的修正和深入主要参考各平台大佬的文章,其中也可能含有少量的个人实验自证,所有代码均优先参考最佳性能。
🤭结语:如果有帮到你的地方,就点个赞关注一下呗,谢谢🎈🎄🌷!!!
🌈【C++】秋招&实习面经汇总篇


文章目录

    • 基础题目
      • 二分查找
        • 基础知识
        • 相关例题
      • 快速排序(模糊划分)
        • 基础知识
        • 相关例题
      • 移除元素(快慢指针)
      • 两数之和(带哈希缓存的查找)
      • 三数之和(排序+三指针)
      • 四数相加II(unordered_map的使用)
      • 有序数组的平方(双端指针)
    • 参考博客


😊点此到文末惊喜↩︎

基础题目

二分查找

基础知识
  1. 适应场景:有序无重复的数组
    • 有序:一次比较即可确定需要查找的另一半,效率高的关键
    • 无重复:一旦有重复元素,使用二分查找法返回的元素下标可能不是唯一,需要进行左右的循环查找
    • 数组:可以进行随机存取
  2. 细节
    • 快速防溢出的除2:mid = L + ((R - L)>>1)
      • 防溢出:如果L 和 R太大,直接相加就有溢出的可能
      • 右移:等价于除法算法,但是效率高
    • 使用前闭后闭的二分区域查找,可以在查找target位置后再进行相同元素相连区域的定位操作。
  3. leetcode题目:704. 二分查找
    // *****************前闭后闭的基本二分查找,可以代替下一种*******************
    int search(vector<int>& nums, int target) {
    // 0. 健壮性检查if(nums.size() <= 0) return -1;
    // 1. 定义边界指针(指向遍历数组区域的边界位置)int left = 0;int right = nums.size() - 1; // 定义target在左闭右闭的区间里
    // 2. 基本算法步骤的循环    while (left <= right) { // 前闭后闭用<=// 划分中间int mid = left + ((right - left)>>1);	// 防止溢出 等同于(left + right)/2if (target < nums[mid]) {				// 目标值在左区间right = mid - 1; } else if (target > nums[mid]) {		// 目标值在右区间left = mid + 1; } else {								// 找到目标值,即相等时return mid; 						// 数组中找到目标值,直接返回下标}}
    // 3. 添加进行左右边界的定位操作// ...return left;// left为相等值未找到时应插入的位置,也可使用-1表示
    }
    
相关例题
  1. leetcode题目:35. 在排序数组中查找元素的第一个和最后一个位置
    • 先通过基本二分法查找目标元素出现的位置
    • 然后使用while(边界判断 && 值判断)获得target值的区间
    ···	// 基本二分查找
    // 寻找相似相邻区间的左右边界
    int l = mid;
    int r = mid;
    if(nums[mid] != target){return res;
    }else{while (vec[left] == target && left >= 0) {--left;}while (vec[right] == target && right <= vec.size() - 1) {++right;}
    }
    res[0]=l;
    res[1]=r;
    return res;
    }
    
    1. 搜索旋转排序数组
    • 整数数组 nums 按升序排列,但是可以进行循环移位,然后进行target的查找
    • 使用时间复杂度为 O(log n) 的算法
      在这里插入图片描述
    • 思路
      • nums[0] <= nums[mid](0 - mid不包含旋转)且nums[0] <= target <= nums[mid] 时 high 向前规约;
      • nums[mid] < nums[0](0 - mid包含旋转),target <= nums[mid] < nums[0] 时向前规约(target 在旋转位置到 mid 之间)
      • nums[mid] < nums[0],nums[mid] < nums[0] <= target 时向前规约(target 在 0 到旋转位置之间)
      • 其他情况向后规约
      • nums[mid] < nums[0],nums[0] > target,target > nums[mid] 三项均为真或者只有一项为真时向后规约
int search(vector<int>& nums, int target) {int lo = 0, hi = nums.size() - 1;while (lo < hi) {int mid = (lo + hi) / 2;if ((nums[0] > target) ^ (nums[0] > nums[mid]) ^ (target > nums[mid]))lo = mid + 1;elsehi = mid;}return lo == hi && nums[lo] == target ? lo : -1;
}
  1. 正序数组查找第k小(二分查找)
    • 给定两个大小分别为 m 和 n 的正序(从小到大)数组 nums1 和 nums2。请你找出并返回这两个正序数组的 中位数
    • 算法的时间复杂度应该为 O(log (m+n))
class Solution {
public:int findKthElm(vector<int>& nums1,vector<int>& nums2,int k){assert(1<=k&&k<=nums1.size()+nums2.size());int le=max(0,int(k-nums2.size())),ri=min(k,int(nums1.size()));while(le<ri){int m=le+(ri-le)/2;if(nums2[k-m-1]>nums1[m]) le=m+1;//这为什么只写一个条件?!因为这是二分的排除条件,不是目标的符合条件,相当于排除条件,最后的结果才是符合条件的结果else ri=m;}//循环结束时的位置le即为所求位置,第k小即为max(nums1[le-1]),nums2[k-le-1]),但是由于le可以为0、k,所以//le-1或者k-le-1可能不存在所以下面单独判断下int nums1LeftMax=le==0?INT_MIN:nums1[le-1];int nums2LeftMax=le==k?INT_MIN:nums2[k-le-1];return max(nums1LeftMax,nums2LeftMax);}double findMedianSortedArrays(vector<int>& nums1, vector<int>& nums2) {int n=nums1.size()+nums2.size();if(n&1){//两个数组长度和为奇数return findKthElm(nums1,nums2,(n>>1)+1);}else{//为偶数return (findKthElm(nums1,nums2,n>>1)+findKthElm(nums1,nums2,(n>>1)+1))/2.0;}}
};

快速排序(模糊划分)

基础知识
  1. 基本思想
    • 选择基准:在待排序列中,按照某种方式挑出一个元素,作为 “基准”(pivot)
    • 分割操作:以该基准在序列中的实际位置,把序列分成两个子序列。此时,在基准左边的元素都比该基准小,在基准右边的元素都比基准大
    • 递归地对两个序列进行快速排序,直到序列为空或者只有一个元素。
  2. 特点
    • 不产生有序子序列,但每次排序后会将基准元素放到最终位置上
    • 每次排序划分子区间越相近越能发挥快排优势
    • 每次可将无序线性表划分成小值区pivot大值区
  3. 算法
int partition(vector<int> &vec, int left, int right) {// 将第一个元素随机化,避免有序数组导致的划分失衡int idx = left + rand() % (right - left + 1);swap(vec[left], vec[idx]);// 初始化:划分元素的位置和值int pos = left;int pivot = vec[left];// 算法部分while (left < right) {// 从后向前找 小于 基准元素的while (vec[right] >= pivot && left < right) --right;// 从前向后找 大于 基准元素的while (vec[left] <= pivot && left < right) ++left;swap(vec[left], vec[right]);}swap(vec[left], vec[pos]);return left;	
}void QuickSort(vector<int> &vec, int left, int right) {if (left > right) return ;int pivot = partition(vec, left, right);QuickSort(vec, left, pivot-1);QuickSort(vec, pivot+1, right);
}

在这里插入图片描述

相关例题
    1. 数组中的第K个最大元素
    • 题目:给定整数数组 nums 和整数 k,请返回数组中第 k 个最大的元素。
    • 思路
      • 快排划分:时间复杂度:O(N) 空间复杂度:O(1)
      • 堆求解:时间复杂度:O(Nlog⁡K) 空间复杂度:O(K)
// 快排划分 
int findKthLargst(vector<int>& nums, int k) {// 划分函数(key:从大到小)auto partition = [&](int left, int right)->int{// 随机化:避免划分失败int idx = left + rand() % (right-left+1);swap(nums[left], nums[idx]);// 划分元素的位置和值int pos = left;int pivot = nums[left];while (left < right) {while (nums[right] <= pivot && left < right) --right;while (nums[left] >= pivot && left < right) ++left;swap(nums[left], nums[right]);}// 划分转移swap(nums[left], nums[pos]);return left;};// 算法int left = 0;int right = nums.size()-1;// 找到正序数组中的第k大while (left <= right) {int mid = partition(left, right);if (k == mid+1) {return nums[mid];} else if (k > mid+1) {left = mid+1;} else {right = mid-1;}}return 0;
}
// 堆处理
int findKthLargest(vector<int>& nums, int k) {priority_queue<int> pq;for (auto &i : nums) {pq.push(i);}for (int i = 0; i < k-1; ++i) {pq.pop();}return pq.top();
}

移除元素(快慢指针)

  1. leetcode题目:27. 移除元素
    • 要求:使用快慢指针以O(n)的时间复杂度和O(1)的空间复杂度进行处理
    • 注意点
      • 快指针fast用于条件判断,慢指针slow用于位置保存
      • 尽量使用for循环避免结尾迭代条件的忘记
    int removeElement(vector<int>& nums, int val) {int slow = 0, fast = 0;     // 定义快慢指针for (; fast < nums.size(); ++fast) {// 快指针进行判断判断if (nums[fast] != val) {nums[slow] = nums[fast];++slow;}}nums.resize(slow);return slow;
    }
    

两数之和(带哈希缓存的查找)

  1. leetcode1. 两数之和
  2. 思路
    • 每次获取一个元素,先判断是否成功,如果不成功则将元素放入哈希缓存表中
    vector<int> twoSum(vector<int>& nums, int target) {vector<int> res;unordered_map<int, int> umap;for (int i = 0; i < nums.size(); ++i) {auto itr = umap.find(target-nums[i]);if (itr != umap.end()) {res.emplace_back(i);res.emplace_back(itr->second);}umap[nums[i]] = i;}return res;
    }
    

三数之和(排序+三指针)

  1. 15. 三数之和
    • 给你一个整数数组 nums ,判断是否存在三元组 [nums[i], nums[j], nums[k]] 满足 i != j、i != k 且 j != k ,同时还满足 nums[i] + nums[j] + nums[k] == 0 。请你返回所有和为 0 且不重复的三元组。
    • 答案中不可以包含重复的三元组。
  2. 思路
    • 固定一个指针进行问题的降维,然后另外两个指针进行区间的运算
      请添加图片描述
vector<vector<int>> threeSum(vector<int>& nums) {const int len = nums.size();vector<vector<int>> res;sort(nums.begin(), nums.end());// 健壮性检查 if (nums.size() < 3 || nums[0] > 0 || nums[nums.size()-1] < 0)return res;// 算法部分// 找出a + b + c = 0// a = nums[i], b = nums[left], c = nums[right]for (int i = 0; i < nums.size(); i++) {// 排序后若第一个元素大于零,则表示后面元素不可能凑成三元组if (nums[i] > 0) return res;// 正确去重方法:比较i和i+1会遗漏掉第一个元素作为首元素的情况if (i > 0 && nums[i] == nums[i - 1]) continue;// 确定剩余两个元素的区间int left = i + 1;int right = nums.size() - 1;while (left < right) {if (nums[i] + nums[left] + nums[right] > 0) right--;else if (nums[i] + nums[left] + nums[right] < 0) left++;else {// 压入第一个,然后对后面的去重res.push_back(vector<int>{nums[i], nums[left], nums[right]});// 去重逻辑应该放在找到一个三元组之后,对b 和 c去重while (right > left && nums[right] == nums[right - 1]) right--;while (right > left && nums[left] == nums[left + 1]) left++;// 找到答案时,双指针同时收缩right--;left++;}}}return res;
}
  1. 注意点
    • 去重操作
      • if (nums[i] == nums[i + 1]) continue;会导致[-1,-1,2]情况的遗漏
      • if (i > 0 && nums[i] == nums[i - 1]) continue;优先判断左部分,不会影响右部分

四数相加II(unordered_map的使用)

  1. leetcode题目:四数相加
    • 记录去重常数级查找通过unordered_set解决
      在这里插入图片描述
    int fourSumCount(vector<int>& A, vector<int>& B, vector<int>& C, vector<int>& D) {unordered_map<int, int> umap; //key:a+b的数值,value:a+b数值出现的次数// 遍历大A和大B数组,统计两个数组元素之和,和出现的次数,放到map中for (int a : A) {for (int b : B) {umap[a + b]++;}}int count = 0; // 统计a+b+c+d = 0 出现的次数// 在遍历大C和大D数组,找到如果 0-(c+d) 在map中出现过的话// 就把map中key对应的value也就是出现次数统计出来。for (int c : C) {for (int d : D) {int val = -(c+d);if (umap.count(val) > 0) {count += umap[val];}}}return count;
    }
    

有序数组的平方(双端指针)

  1. leetcode题目:977. 有序数组的平方
    在这里插入图片描述
  2. 思路:
    • 正常思路:从0开始向中间 + 辅助空间
    • 逆向思路:left和right两个指针分别从两端向中间进行夹逼
vector<int> sortedSquares(vector<int>& nums) {// 健壮性检查// 初始化const int len = nums.size();int left = 0;int right = len-1;vector<int> res(len);// 算法部分for (int index = len-1; left <= right; --index) {if (nums[left] * nums[left] > nums[right] *nums[right]) {res[index] = nums[left] * nums[left];++left;} else {res[index] = nums[right] *nums[right];--right;}}return res;
}


少年,我观你骨骼清奇,颖悟绝伦,必成人中龙凤。
不如点赞·收藏·关注一波

🚩点此跳转到首行↩︎

参考博客

  1. 「代码随想录」
  2. codetop前200
  3. 力扣(LeetCode)
  4. 旋转数组——极简 Solution

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

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

相关文章

视觉检测系统,外观细节无可挑剔

在传统行业中&#xff0c;利用人工检测来检测产品外观缺陷依然是主流&#xff0c;但由于竞争的加剧&#xff0c;对企业生产效率的要求也越来越高。传统的检测产品外观缺陷问题的方法就是透过人工目检&#xff0c;或者工人采用游标卡尺等工具检测&#xff0c;此种方式检测速度慢…

振弦式应变计:简单操作,方便实用的应变监测工具

在现代工程领域中&#xff0c;对于结构物的应变监测是一项至关重要的任务。振弦式应变计作为一种高精度、高稳定性的应变监测工具&#xff0c;因其简单操作、方便实用的特点&#xff0c;受到了广大工程师和技术人员的青睐。 振弦式应变计的工作原理基于振弦的振动特性。它通过将…

HCIA笔记

console 登录设备的特点&#xff1a; 1、带外&#xff0c;不依赖网络本身的连通性。 2、独占&#xff0c;console口不能被多人同时使用&#xff0c;具备唯一性。 3、本地&#xff0c;console口长度有限&#xff0c;一般只能在机房或者设备现场来使用。 4、只能实现命令行的管理…

ngAlain下使用nz-select与文件上传框出现灵异bug

bug描述 初始化页面&#xff0c;文件上传框无法出现&#xff1a; 但点击一次选择框以后&#xff0c;就会出现&#xff1a; 真的很神奇。。。 下面逐步排查看看是什么原因。 设想一&#xff1a; 选择框与文件框不可同时存在&#xff0c;删掉选择框看看&#xff1a; 还…

隐语SecretFlow实训营-第8讲:快速上手隐语SCQL的开发实践

SCQL使用/集成实践 目前SCQL只开放API供用户使用/集成 使用SCDBClient上手体验可以基于SCQL API开发封装白屏产品&#xff0c;或集成到业务链路中 使用流程&#xff1a; 部署系统 环境配置&#xff1a; 机器配置&#xff1a;CPU/MEM最低8C16G机构之间的网络互通 镜像&…

全面了解海外网络专线

SD-WAN海外网络专线技术 在选择海外网络专线服务时&#xff0c;企业需要考虑多个因素&#xff0c;包括服务商的可靠性、价格、技术支持和合规性。本文将探讨跨境网络专线的价格因素、合法跨境上网的方式&#xff0c;以及SD-WAN跨境上网专线的优势。 跨境网络专线的价格与办理…

STM32学习笔记(11_2)- W25Q64简介和工作原理

无人问津也好&#xff0c;技不如人也罢&#xff0c;都应静下心来&#xff0c;去做该做的事。 最近在学STM32&#xff0c;所以也开贴记录一下主要内容&#xff0c;省的过目即忘。视频教程为江科大&#xff08;改名江协科技&#xff09;&#xff0c;网站jiangxiekeji.com 本期学…

Golang Gin框架

1、这篇文章我们简要讨论一些Gin框架 主要是给大家一个基本概念 1、Gin主要是分为路由和中间件部分。 Gin底层使用的是net/http的逻辑&#xff0c;net/http主要是说&#xff0c;当来一个网络请求时&#xff0c;go func开启另一个协程去处理后续(类似epoll)。 然后主协程持续…

蓝桥杯物联网竞赛_STM32L071_15_ADC/脉冲模块

ADC模块用的是RP1不用多说了&#xff0c;主要是脉冲模块&#xff0c;这个模块没考过 这个脉冲模块放出脉冲&#xff0c;主要能用TIM捕获到这个脉冲的高电平持续时间即可 CubMx配置&#xff1a; 脉冲模块的引脚与PB0相连&#xff0c;所以用PB0读取上升沿记的数和下降沿记的数&am…

炫我科技:云渲染领域的佼佼者

随着数字化时代的来临&#xff0c;云渲染技术正逐渐成为影视、游戏、动画等创意产业的重要支柱。在这一领域中&#xff0c;炫我科技凭借其卓越的技术实力、优质的服务以及不断创新的精神&#xff0c;已然成为了云渲染行业的佼佼者。 炫我科技自成立之初&#xff0c;便以打造高…

tkinter实现通用对账文件解析软件

软件需求 和银行等金融机构合作过程中&#xff0c;经常会有还款计划、放款文件等定时推送的文件&#xff0c;以常见的分隔符进行分隔开&#xff0c;为了在系统出现问题时&#xff0c;快速查找异常数据&#xff0c;写了一个小工具解析这些文件并写入到excel中。 软件功能 将常…

Vue ElementPlus Input 输入框

Input 输入框 通过鼠标或键盘输入字符 input 为受控组件&#xff0c;它总会显示 Vue 绑定值。 通常情况下&#xff0c;应当处理 input 事件&#xff0c;并更新组件的绑定值&#xff08;或使用v-model&#xff09;。否则&#xff0c;输入框内显示的值将不会改变&#xff0c;不支…

异构加速GPU服务器设计方案:904-全国产化异构加速GPU服务器

全国产化异构加速GPU服务器 一、产品介绍 X7340H0是中科可控基于HYGON系列处理器开发的一款全新高端2U双路GPU服务器。X7340H0采用优异的可扩展架构设计&#xff0c;支持高密度扩展GPU加速卡&#xff0c;为深度学习推理场景提供更加安全可靠、高性价比的解决方案。 性能卓越 ●…

经典文献阅读之--LOG-LIO(高效局部几何信息估计的激光雷达惯性里程计)

0. 简介 局部几何信息即法线和点分布在基于激光雷达的同时定位与地图构建&#xff08;SLAM&#xff09;中是至关重要&#xff0c;因为它为数据关联提供了约束&#xff0c;进一步确定了优化方向&#xff0c;最终影响姿态的准确性。然而即使在使用KD树或体素图的辅助下&#xff…

【CANN训练营笔记】AscendCL图片分类应用(C++实现)

样例介绍 基于PyTorch框架的ResNet50模型&#xff0c;对*.jpg图片分类&#xff0c;输出各图片所属分类的编号、名称。 环境介绍 华为云AI1s CPU&#xff1a;Intel Xeon Gold 6278C CPU 2.60GHz 内存&#xff1a;8G NPU&#xff1a;Ascend 310 环境准备 下载驱动 wget ht…

在 Windows 中安装部署并启动连接 MongoDB 7.x(命令行方式启动、配置文件方式启动、将启动命令安装为系统服务实现开机自启)

MongoDB 的下载 下载地址&#xff1a;https://www.mongodb.com/try/download/community 这里需要对 MongoDB 的版本号说明一下&#xff1a; MongoDB 版本号的命名规则是 x.y.z&#xff0c;当其中的 y 是奇数时表示当前的版本为开发版&#xff0c;当其中的 y 是偶数时表示当前的…

非关系型数据库之Redis配置与优化

一、关系数据库与非关系型数据库 1.1关系型数据库 关系型数据库是一个结构化的数据库&#xff0c;创建在关系模型&#xff08;二维表格模型&#xff09;基础上一般面向于记录。SQL语句&#xff08;标准数据查询语言&#xff09;就是一种基于关系型数据库的语言&#xff0c;用…

首场直播,就在4月11日!

2024年的第一场直播&#xff0c;我们把目光聚焦到“大会员”。 这一次我们想聊聊&#xff0c;当大会员遇上泛零售企业&#xff0c;会产生怎样的“火花”。泛零售企业突破增长压力的机会在哪里&#xff1f;又有哪些挑战必须直面&#xff1f; 本次直播将结合泛零售企业“多业态、…

GEE问题——在使用sentienl数据云掩膜的时候发现出现中间连贯性的“条带”问题,如何解决?

简介 在使用sentienl+landsat数据掩膜的时候发现出现了中间连贯性的条带问题,如何解决?这里我们使用GEE出品的Landsat和sentinel数据的过程中,当我们进行云掩膜的时候出现了条带的问题。 问题 您注意到这个问题了吗? 我该如何消除它们(例如,在镶嵌前遮蔽瓦片最外层的 …

1379. 找出克隆二叉树中的相同节点

说在前面 &#x1f388;不知道大家对于算法的学习是一个怎样的心态呢&#xff1f;为了面试还是因为兴趣&#xff1f;不管是出于什么原因&#xff0c;算法学习需要持续保持。 题目描述 给你两棵二叉树&#xff0c;原始树 original 和克隆树 cloned&#xff0c;以及一个位于原始…