剑指Offer(数据结构与算法面试题精讲)C++版——day4
剑指Offer(数据结构与算法面试题精讲)C++版——day4
- 题目一:和为k的子数组
- 题目二:0和1个数相同的子数组
- 题目三:左右两边子数组的和相等
题目一:和为k的子数组

结合前面着重阐述的双指针法这一经典的算法技巧,我们在面对当前的问题时,很自然且清晰地能够联想到运用双指针来进行处理。于是,我们设置了左指针 p 和右指针 q ,让它们从数组的最左端开始,逐步向右进行遍历操作。在这个过程中,由指针 p 和指针 q 之间所涵盖的数组元素共同构成了一个子数组。算法的核心逻辑在于对这个子数组的和进行判断与处理。当子数组的和小于给定的数值 k 时,为了使子数组的和能够更接近或达到 k ,我们选择将指针 q 朝着右方移动,从而纳入更多的数组元素,期望增加子数组的和;而当子数组的和大于 k 时,为了减小子数组的和,我们会将指针 p 向右移动,剔除部分数组元素。然而,这种看似简洁高效的方法却存在着一个不容忽视的局限性。该方法仅仅适用于数组元素均为正整数的情况。因为如果数组中存在非正整数,当执行向右移动指针的操作时,很可能会出现子数组的和变得更小的情况,这样一来,原本通过移动指针来增加子数组和值的目的就无法达成了,也就导致这种方法在处理包含非正整数的数组时失去了有效性。
因此,需要考虑使用其他方法,如果是直接使用蛮力法,那么需要借助于数组下标,差分出O(n^2)种左右不同边界,对于每一个子数组,为了计算和,还需要O(n)时间,因此蛮力法的时间复杂度为O(n^3),还是比较高的。分析发现,可以借助于数列和的特点来优化时间复杂度,也是一种空间换时间的方法,记Si表示数组从下标0-i之间的所有元素的和,那么在定界之后,比如说子数组的左右边界对应在数组中的下标为i和j,那么这段子数组的和值即为Sj-Si。这样得到的时间复杂度为O(n^2),最终得到的代码如下:
# include <iostream>
# include <algorithm>
using namespace std;
int findChildArrCount(int arr[],int len,int k) {int count=0,sum=0;int arrSum[len]= {0};for(int i=0; i<len; ++i) {sum+=arr[i];arrSum[i]=sum;}for(int i=0; i<len; ++i) {for(int j=i; j<len; ++j) {if((i==0&&arrSum[j]==k)||((arrSum[j]-arrSum[i-1])==k)) {count++;}}}return count;
}
int main() {int arr[]= {1,1,1};int len=sizeof(arr)/sizeof(arr[0]);int k=2;cout<<"这样的数组个数为:"<<findChildArrCount(arr,len,k);return 0;
}

题目二:0和1个数相同的子数组

我们能够从前面题目一中关于子数列的和的相关内容获取灵感,从而想到一个简洁而有效的思路。就像之前处理问题那样,我们还是聚焦于计算数组中特定元素的和,这里着重考虑的是与统计偶数个数相关的和值计算。具体来说,我们设定
Si为数组从起始下标 0 到下标 i 这一区间内所有元素的总和,通过这种方式来构建一个便于后续分析的和值序列。
当我们对数组进行定界操作时,会确定子数组的左右边界,不妨设子数组的左边界在数组中的下标为 i ,右边界下标为 j 。基于前面定义的 Si,我们可以清晰地得出,这个子数组的和值能够通过 Sj−Si 计算出来。
接下来,我们需要对数组进行第二次遍历。在这次遍历过程中,我们会仔细检查定界指针 i 和 j 之间的子数组的元素和情况。这里存在两个关键的判断条件,缺一不可。一方面,定界指针 i 和 j 之间的子数组元素和必须恰好等于 (j−i+1)/2 ,这是由于我们要寻找的子数组中 0 和 1 的数量相等,所以其元素和必然是子数组长度的一半。另一方面,该子数组的长度必须是偶数,因为只有长度为偶数的情况下,才有可能实现 0 和 1 的个数相等这一目标。只有当这两个条件同时满足时,才能判定这样的子数组符合我们预先设定的要求。
这种解决问题的方法在时间和空间复杂度上都具有明显的优势,其开销相对较小。从具体的操作流程来看,仅仅需要对数组进行两次遍历即可。第一次遍历主要是对数组元素进行求和统计,为后续的判断提供必要的数据支撑;第二次遍历则是在第一次遍历所得和值的基础上,逐一检查每个子数组的和是否恰好等于数组长度的一半,以此来筛选出符合条件的子数组。基于以上完整的思路,我们最终能够编写出相应的代码。
# include <iostream>
# include <algorithm>
using namespace std;
int findMaxLength(int arr[],int len) {int sum=0,maxLen=0;bool condition1,condition2;int arrSum[len]= {0};for(int i=0; i<len; ++i) {sum+=arr[i];arrSum[i]=sum;}for(int i=0; i<len; ++i) {for(int j=i; j<len; ++j) {condition1=(j-i+1)%2==0;//子数组的个数为偶数condition2=(i==0&&arrSum[j]==j/2)||(arrSum[j]-arrSum[i-1]==(j-i+1)/2);//满足和为个数的一半 if(condition1&&condition2&&(j-i+1)>maxLen) {//满足上述2个条件且超过当前最大子数组长度 maxLen=j-i+1;}}}return maxLen;
}
int main() {int arr[]= {0,1,0,1};int len=sizeof(arr)/sizeof(arr[0]);cout<<"这样的数组最大长度为:"<<findMaxLength(arr,len);return 0;
}

题目三:左右两边子数组的和相等

结合前面两道题我们会发现,这种方法还特别好用,利用空间换时间。我们可以立即想到一个思路,第一次遍历数组,统计Si,那么对于给定的一个枢轴元素取为arr[k],我们只需要比较S(k-1)-S(i-1) 和S(len-1)-S(k)是否相等即可。
这样需要消耗O(n)的空间复杂度,分析发现,其实我们只需要去统计整个数组的和(记为sum),以及前Si,那么后半段的和可以利用sum-arr[k]-S(k-1)拿到,这样空间复杂度就优化成了O(1)了。最终得到的代码如下:
# include <iostream>
# include <algorithm>
using namespace std;
int findPivotIndex(int arr[],int len) {int sum=0,tmpSum=0;for(int i=0; i<len; ++i) {//统一次总和sum+=arr[i];}for(int i=0; i<len-1; ++i) {tmpSum+=arr[i];//枢轴左半边的和if(i==0||i==len-1)continue;//排除枢轴在边界的情况if(tmpSum==sum-arr[i+1]) {return i;}}return -1;
}
int main() {int arr[]= {1,7,3,6,2,9};int len=sizeof(arr)/sizeof(arr[0]);cout<<"枢轴元素的下标为:"<<findPivotIndex(arr,len);return 0;
}
这里补充两点说明:
(1)题目三中对于子数组,可能有些OJ平台会考虑到空数组的情况,需要根据实际判题结果来调整,这里是以子数组不能为空来编写的;
(2)需要注意一点,对于数组取和,最好使用long long来存储,可能有些测试数据的和值超过了INT的最大范围。
我是【Jerry说前后端】,本系列精心挑选的算法题目全部基于经典的《剑指 Offer(数据结构与算法面试题精讲)》。在如今竞争激烈的技术求职环境下,算法能力已成为前端开发岗位笔试考核的关键要点。通过深入钻研这一系列算法题,大家能够系统地积累算法知识和解题经验。每一道题目的分析与解答过程,都像是一把钥匙,为大家打开一扇通往高效编程思维的大门,帮助大家逐步提升自己在数据结构运用、算法设计与优化等方面的能力。
无论是即将踏入职场的应届毕业生,还是想要进一步提升自己技术水平的在职开发者,掌握扎实的算法知识都是提升竞争力的有力武器。希望大家能跟随我的步伐,在这个系列中不断学习、不断进步,为即将到来的前端笔试做好充分准备,顺利拿下心仪的工作机会!快来订阅吧,让我们一起开启这段算法学习之旅!
相关文章:
剑指Offer(数据结构与算法面试题精讲)C++版——day4
剑指Offer(数据结构与算法面试题精讲)C版——day4 题目一:和为k的子数组题目二:0和1个数相同的子数组题目三:左右两边子数组的和相等 题目一:和为k的子数组 结合前面着重阐述的双指针法这一经典的算法技巧&…...
从代码学习深度学习 - NLP之文本预处理 PyTorch版
文章目录 前言1. 文本预处理理论知识1.1 文本清洗与标准化1.2 分词(Tokenization)1.3 词频统计与词汇表构建1.4 序列表示与批次生成1.5 预处理的意义2. 文本预处理的核心代码解析2.1 读取数据集:`read_time_machine`2.2 分词处理:`tokenize`2.3 词频统计:`count_corpus`2.…...
WebRTC技术简介及应用场景
写在前面 本文是参考稀土掘金的文章,整理得出,版权归原作者所有!参考链接请点击跳转 WebRTC(Web Real-Time Communication) 是一项开源技术,允许浏览器和移动应用直接进行实时音视频通信和数据传输,无需安装插件或第三方软件。它…...
介绍几种创意登录页(含完整源码)
今天为大家收集了几种不同风格的登录页,搭配动态渐变背景,效果绝对惊艳! CSS3实现动态渐变玻璃拟态登录页 一、开篇语 纯CSS实现当下最火的玻璃拟态(Morphism)风格登录页,搭配动态渐变背景,效果绝对惊艳! …...
git分布式控制工具详解
1. 版本控制器的方式 1.1 集中式版本控制工具 特点: 版本库集中存放在中央服务器必须联网才能工作(局域网/互联网)个人修改后提交到中央版本库 举例:SVN、CVS 1.2 分布式版本控制工具 特点: 无"中央服务器&qu…...
Uni-app入门到精通:uni-app的基础组件
1、view view是容器组件,类似于HTML中的<div></div>标签,用于包裹各种元素内容,是页面布局常用的组件。view组件的属性如下 属性类型默认值说明hover-classStringnone指定按下去的样式类。当hover-class"none"时&…...
R语言从专家到小白
文章目录 下载安装R下载安装R StudioCRAN 下载安装R Index of /bin https://cran.r-project.org/ 下载安装R Studio https://posit.co/download/rstudio-desktop/ CRAN R综合档案网络。 CRAN 镜像是一个提供 R 语言软件和包的在线服务,用户可以从不同的地区选择…...
显示器工艺简介
华星光电显示器的生产工艺流程介绍,从入厂原料到生产出显示器的整体工艺介绍 华星光电显示器的生产工艺流程主要包括以下几个阶段,从原材料入厂到最终显示器的生产: 原材料准备 玻璃基板:显示器的核心材料,通常采用超…...
大文件上传源码,支持单个大文件与多个大文件
大文件上传源码,支持单个大文件与多个大文件 Ⅰ 思路Ⅱ 具体代码前端--单个大文件前端--多个大文件前端接口后端 Ⅰ 思路 具体思路请参考我之前的文章,这里分享的是上传流程与源码 https://blog.csdn.net/sugerfle/article/details/130829022 Ⅱ 具体代码…...
C语言--插入排序
插入排序:简单而高效的排序算法 在计算机科学中,排序是一种常见的操作,用于将一组数据按照特定的顺序排列。插入排序(Insertion Sort)是一种简单直观的排序算法,它的工作原理类似于我们整理扑克牌的过程。…...
L2-024 部落 #GPLT,并查集 C++
文章目录 题目解读输入格式输出格式 思路Ac Code参考 题目解读 我们认为朋友的朋友都算在一个部落里,于是要请你统计一下,在一个给定社区中,到底有多少个互不相交的部落?并且检查任意两个人是否属于同一个部落。 输入格式 第一…...
前端面试题(三):axios有哪些常用的方法
Axios 是一个基于 Promise 的 HTTP 客户端,用于浏览器和 Node.js 中发送 HTTP 请求。它提供了一些常用的方法来处理不同类型的请求。以下是 Axios 中常用的一些方法: 1. axios.get() 用于发送 GET 请求,从服务器获取数据。 axios.get(/api/d…...
JSON 基础知识(一)
第一部分:JSON 基础知识 📢 快速掌握 JSON!文章 视频双管齐下 🚀 如果你觉得阅读文章太慢,或者更喜欢 边看边学 的方式,不妨直接观看我录制的 JSON 课程视频!🎬 视频里会用更直观…...
SSM框架学习(Day-1)
1.spring系统架构 自底而上进行,上层依赖于下层,首先最底层是Core Container -- 核心容器, 再往上是AOP(面向切面编程)和Aspects(AOP)思想的实现, 我个人的理解是, 它可以在不惊动你原始程序的基础上, 给它增强功能,类似于反射;再往上是数据访问层。 C…...
使用 PyTorch 的 `GradualWarmupScheduler` 实现学习率预热
使用 PyTorch 的 GradualWarmupScheduler 实现学习率预热 在深度学习中,学习率(Learning Rate, LR)是影响模型训练效果的关键超参数之一。为了提升模型的收敛速度和稳定性,学习率调度策略变得尤为重要。其中,学习率预热(Learning Rate Warmup) 是一种常用的策略,它通过…...
Redis 中 Set(例如标签) 和 ZSet(例如排行榜) 的详细对比,涵盖定义、特性、命令、适用场景及总结表格
以下是 Redis 中 Set 和 ZSet 的详细对比,涵盖定义、特性、命令、适用场景及总结表格: 1. 核心定义 数据类型SetZSet(Sorted Set)定义无序的、唯一的字符串集合,元素不重复。有序的、唯一的字符串集合,每个…...
在线记事本——支持Markdown
项目地址 https://github.com/Anyuersuper/CloudNotebook 百度网盘 通过网盘分享的文件:CloudNotebook-master.zip 链接: https://pan.baidu.com/s/1_Y--aBzNkKiFRIMHYmwPdA?pwdyuer 提取码: yuer 📝 云笔记 (Cloud Notebook) 云笔记是一个简洁、安全…...
C# 中充血模型和贫血模型
在C#中,充血模型(Rich Domain Model)和贫血模型(Anemic Domain Model)是两种截然不同的领域建模方式,核心区别在于业务逻辑的归属。以下是通俗易懂的解释: 1. 贫血模型ÿ…...
Java技术生态前沿洞察:虚拟线程引领并发革命,框架创新赋能云原生时代
Java技术生态正迎来新一轮变革浪潮。虚拟线程的落地成为高并发编程范式转折点,其极低资源开销特性在电商秒杀场景中展现出3倍吞吐量提升,彻底改写传统线程模型性能边界。Spring Boot 3.2原生支持虚拟线程,结合Observation API与HTTP客户端优化…...
Day2:前端项目uniapp壁纸实战
先来做一个轮番图。 效果如下: common-style.css view,swiper,swiper-item{box-sizing: border-box; } index.vue <template><view class"homeLayout"><view class"banner"><swiper circular indicator-dots autoplay…...
人工智能赋能工业制造:智能制造的未来之路
一、引言 随着人工智能技术的飞速发展,其应用场景不断拓展,从消费电子到医疗健康,从金融科技到交通运输,几乎涵盖了所有行业。而工业制造作为国民经济的支柱产业,也在人工智能的浪潮中迎来了深刻的变革。智能制造&…...
V-SHOW和箭头函数在VUE项目的踩坑点
v-show和v-if v-show控制显示隐藏是通过控制CSS的display决定dom节点的显示和隐藏。v-if通过控制dom节点的渲染与否实现元素的显示和隐藏。 在vue中,template标签不参与页面渲染,也不会破坏代码的层级结构,所以多和v-if结合控制元素的显示隐…...
LeetCode Hot100 刷题笔记(3)—— 链表
目录 前言 1. 相交链表 2. 反转链表 3. 回文链表 4. 环形链表 5. 环形链表 II 6. 合并两个有序链表 7. 两数相加 8. 删除链表的倒数第 N 个结点 9. 两两交换链表中的节点 10. K 个一组翻转链表 11. 随机链表的复制 12. 排序链表 13. 合并 K 个升序链表 14. LRU 缓存 前言 一、…...
Spring 概念
Spring 是一个功能强大、灵活且广泛使用的 Java 企业级开发框架,它诞生于 2003 年,由 Rod Johnson 创建,初衷是简化 Java EE 的开发过程。 一、Spring 是什么? 简单来说: Spring 是一个轻量级的 Java 开发框架&#…...
状态机思想编程
1. LED流水灯的FPGA代码 在这个任务中,首先我们会使用状态机的思想来设计一个LED流水灯的控制逻辑。LED流水灯一般需要依次点亮不同的LED,并且循环播放。我们将其分为几个状态,每个状态控制一个或一组LED灯。 状态机设计 假设我们有8个LED…...
第二十八章:Python可视化图表扩展-和弦图、旭日图、六边形箱图、桑基图和主题流图
一、引言 在数据可视化领域,除了常见的折线图、柱状图和散点图,还有一些高级图表类型可以帮助我们更直观地展示复杂数据关系。本文将介绍五种扩展图表:和弦图、旭日图、六边形箱图、桑基图和主题流图。这些图表在展示数据关系、层次结构和流量…...
基于vue框架的重庆美食网站的设计与实现kt945(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。
系统程序文件列表 项目功能:用户,美食分类,美食菜品 开题报告内容 基于Vue框架的重庆美食网站的设计与实现开题报告 一、选题背景与意义 (一)选题背景 重庆,作为中国西南地区的璀璨明珠,以其独特的地理位置和丰富…...
Metal学习笔记十三:阴影
在本章中,您将了解阴影。阴影表示表面上没有光。当另一个表面或对象使对象与光线相遮挡时,您会看到对象上的阴影。在项目中添加阴影可使您的场景看起来更逼真,并提供深度感。 阴影贴图 阴影贴图是包含场景阴影信息的纹理。当光线照射到物体…...
时间梯度匹配损失 TGMLoss
目录 时间梯度匹配损失(Temporal Gradient Matching Loss, TGM Loss) 完整示例,该损失函数常用于视频预测、运动平滑等任务,通过约束预测序列的时间梯度与真实序列一致来提升时序连续性 训练测试demo代码: 时间梯度匹配损失(Temporal Gradient Matching Loss, TGM Los…...
iPhone XR:一代神机,止步于此
什么样的 iPhone ,才配称为一代神机? 我曾经用过iPhone 4S、iPhone 6S Plus、iPhone 8 Plus,iPhone SE2、iPhone XR、iPhone 13、iPhone 14 Plus、iPhone 15/Pro。 不管硬件再怎么卷,不管囊中是否羞涩,主力机基本没考…...
