《剑指 Offer》专项突破版 - 面试题 8 : 和大于或等于 k 的最短子数组(C++ 实现)- 详解同向双指针(滑动窗口算法)

本文主要是介绍《剑指 Offer》专项突破版 - 面试题 8 : 和大于或等于 k 的最短子数组(C++ 实现)- 详解同向双指针(滑动窗口算法),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

前言

一、暴力求解

二、同向双指针(滑动窗口算法)



前言

题目链接:. - 力扣(LeetCode)

题目

输入一个正整数组成的数组和一个正整数 k,请问数组中和大于或等于 k 的连续子数组的最短长度是多少?如果不存在所有数字之和大于或等于 k 的子数组,则返回 0。例如,输入数组 [5, 1, 4, 3],k 的值为 7,和大于或等于 7 的最短连续子数组是 [4, 3],因此输出它的长度 2。

分析

子数组由数组中一个或连续的多个数字组成。一个子数组可以用两个指针表示。如果第 1 个指针 left 指向子数组的第 1 个数字,第 2 个指针 right 指向子数组的最后一个数字,那么子数组就是由这两个指针之间的所有数字组成的


一、暴力求解

  1. 先固定指针 left(最开始指向数组中的第 1 个元素)。

  2. 然后从 left 开始不断向右移动指针 right,直到两个指针之间的子数组中所有数字之和大于或等于 k(子数组的长度为 right - left + 1),或者 right 超出范围,即不存在和大于或等于 k 的子数组。

  3. 如果不存在和大于或等于 k 的子数组,则说明已经尝试了所有的可能性,可以结束这个查找的过程了;否则 ++ left,然后重复步骤 1 和 2,直到 left 超出范围。

这种解法的时间复杂度是 O(n^2)。

当 left = 0 时,right 的最优解为 2(right >= 2 && right < 4,两个指针之间的子数组中所有数字之和大于或等于 k);

当 left = 1 时,right 的最优解为 3;

当 left = 2 时,right 的最优解为 3;

当 left = 3 时,right 没有解。

class Solution {
public:int minSubArrayLen(int target, vector<int>& nums) {int n = nums.size();int minLen = n + 1;for (int left = 0; left < n; ++left){int sum = 0;for (int right = left; right < n; ++right){sum += nums[right];if (sum >= target){if (right - left + 1 < minLen)minLen = right - left + 1;break;}}if (sum < target)break;}return minLen == n + 1 ? 0 : minLen;}
};


二、同向双指针(滑动窗口算法)

可以对上述解法进行优化。指针 left 和 right 初始化的时候指向数组的第 1 个元素

  1. 不断向右移动指针 right,直到两个指针之间的子数组数字之和大于或等于 k(子数组的长度为 right - left + 1)。

  2. 停止右移指针 right,转而不断向右移动指针 left,直到两个指针之间的子数组数字之和小于 k。注意:指针 left 每向右移动一步,如果两个指针之间的子数组数字之和大于或等于 k,那么就要更新最短子数组的长度,这相当于利用了第 1 步产生的结果

  3. 重复步骤 1 和 2,直到 right 超出范围。

class Solution {
public:int minSubArrayLen(int target, vector<int>& nums) {int n = nums.size();int minLen = n + 1;int left = 0;int sum = 0;for (int right = 0; right < n; ++right){sum += nums[right];while (sum >= target){if (right - left + 1 < minLen)minLen = right - left + 1;sum -= nums[left];++left;}}return minLen == n + 1 ? 0 : minLen;}
};

尽管上述代码中有两个嵌套的循环,该解法的时间复杂度仍然是 O(n)。这是因为在这两个循环中,变量 left 和 right 都是只增加不减少,变量 right 从 0 增加到 n - 1,变量 left 从 0 最多增加到 n - 1,因此总的执行次数是 O(n)

这篇关于《剑指 Offer》专项突破版 - 面试题 8 : 和大于或等于 k 的最短子数组(C++ 实现)- 详解同向双指针(滑动窗口算法)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/598705

相关文章

Redis中6种缓存更新策略详解

《Redis中6种缓存更新策略详解》Redis作为一款高性能的内存数据库,已经成为缓存层的首选解决方案,然而,使用缓存时最大的挑战在于保证缓存数据与底层数据源的一致性,本文将介绍Redis中6种缓存更... 目录引言策略一:Cache-Aside(旁路缓存)策略工作原理代码示例优缺点分析适用场景策略二:Re

Flutter实现文字镂空效果的详细步骤

《Flutter实现文字镂空效果的详细步骤》:本文主要介绍如何使用Flutter实现文字镂空效果,包括创建基础应用结构、实现自定义绘制器、构建UI界面以及实现颜色选择按钮等步骤,并详细解析了混合模... 目录引言实现原理开始实现步骤1:创建基础应用结构步骤2:创建主屏幕步骤3:实现自定义绘制器步骤4:构建U

SpringBoot中四种AOP实战应用场景及代码实现

《SpringBoot中四种AOP实战应用场景及代码实现》面向切面编程(AOP)是Spring框架的核心功能之一,它通过预编译和运行期动态代理实现程序功能的统一维护,在SpringBoot应用中,AO... 目录引言场景一:日志记录与性能监控业务需求实现方案使用示例扩展:MDC实现请求跟踪场景二:权限控制与

Android实现定时任务的几种方式汇总(附源码)

《Android实现定时任务的几种方式汇总(附源码)》在Android应用中,定时任务(ScheduledTask)的需求几乎无处不在:从定时刷新数据、定时备份、定时推送通知,到夜间静默下载、循环执行... 目录一、项目介绍1. 背景与意义二、相关基础知识与系统约束三、方案一:Handler.postDel

Java注解之超越Javadoc的元数据利器详解

《Java注解之超越Javadoc的元数据利器详解》本文将深入探讨Java注解的定义、类型、内置注解、自定义注解、保留策略、实际应用场景及最佳实践,无论是初学者还是资深开发者,都能通过本文了解如何利用... 目录什么是注解?注解的类型内置注编程解自定义注解注解的保留策略实际用例最佳实践总结在 Java 编程

MySQL数据库约束深入详解

《MySQL数据库约束深入详解》:本文主要介绍MySQL数据库约束,在MySQL数据库中,约束是用来限制进入表中的数据类型的一种技术,通过使用约束,可以确保数据的准确性、完整性和可靠性,需要的朋友... 目录一、数据库约束的概念二、约束类型三、NOT NULL 非空约束四、DEFAULT 默认值约束五、UN

Python使用Matplotlib绘制3D曲面图详解

《Python使用Matplotlib绘制3D曲面图详解》:本文主要介绍Python使用Matplotlib绘制3D曲面图,在Python中,使用Matplotlib库绘制3D曲面图可以通过mpl... 目录准备工作绘制简单的 3D 曲面图绘制 3D 曲面图添加线框和透明度控制图形视角Matplotlib

使用Python实现IP地址和端口状态检测与监控

《使用Python实现IP地址和端口状态检测与监控》在网络运维和服务器管理中,IP地址和端口的可用性监控是保障业务连续性的基础需求,本文将带你用Python从零打造一个高可用IP监控系统,感兴趣的小伙... 目录概述:为什么需要IP监控系统使用步骤说明1. 环境准备2. 系统部署3. 核心功能配置系统效果展

MySQL中的分组和多表连接详解

《MySQL中的分组和多表连接详解》:本文主要介绍MySQL中的分组和多表连接的相关操作,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录mysql中的分组和多表连接一、MySQL的分组(group javascriptby )二、多表连接(表连接会产生大量的数据垃圾)MySQL中的

Java 实用工具类Spring 的 AnnotationUtils详解

《Java实用工具类Spring的AnnotationUtils详解》Spring框架提供了一个强大的注解工具类org.springframework.core.annotation.Annot... 目录前言一、AnnotationUtils 的常用方法二、常见应用场景三、与 JDK 原生注解 API 的