欢迎来到啾啾的博客🐱。
记录学习点滴。分享工作思考和实用技巧,偶尔也分享一些杂谈💬。
有很多很多不足的地方,欢迎评论交流,感谢您的阅读和评论😄。

博客头像3.0.png

目录

  • 引言
  • 1 双指针:Two Pointers
    • 1.1 左右指针
    • 1.2 滑动窗口

引言

在刷完代码随想录、了解了基本的数据结构后,让我们学算法(时隔四个月了)。
阅读本篇可对基础的双指针算法有深入理解。

简单来说,解决算法问题我们可以遵循一些简单的模板,确认步骤、找合适的数据结构,解。

1 双指针:Two Pointers

  1. 什么是“双指针”模式?它主要想解决什么问题?

双指针核心是减少遍历,优化时间复杂度。
主要是为了将暴力解法中 O(n²) 的时间复杂度(两层循环)优化到 O(n)(一层循环)。它本身就是一种 O(n) 的解法。

  1. “左右指针”和“快慢指针”这两种方法,在使用场景上最大的区别是什么?(比如,一个通常用在什么数据结构上,另一个用在什么上?一个对原始数据有什么要求?)

左右指针主要适用于有序数组(不是非要有序数组),适用于搜索精确值。
快慢指针核心在于利用速度差来解决问题,用来解决数据结构中位置和环路的问题。
最经典的应用场景是链表,而不是数组。

1.1 左右指针

左右指针主要使用与有序数组,也通用于数组。
因为左右指针的条件判断对于位置信息有要求,需要有效移动位置。

LeetCode:
125. 验证回文串
简单的回文字符串判断。利用双指针快速遍历,优化时间复杂度。

11. 盛最多水的容器
明白移动短板是收益最高的选项后,利用双指针移动短板。
这里除了双指正还需要注意的是,我们写解答时,要尽量分开状态处理和状态转移。

  • 错误示例
class Solution {/**移动短板,因为短板没有潜力了*/public int maxArea(int[] height) {int length = height.length;int left = 0;int right =length-1;int currentArea = Math.min(height[left],height[right]) * ( right-left);;while(left < right){if(height[left] < height[right]){left ++;}else{right --;}int tempArea = Math.min(height[left],height[right]) * ( right-left);currentArea = tempArea > currentArea? tempArea:currentArea;}return currentArea;}
}
  • 正确示例
class Solution {public int maxArea(int[] height) {int maxArea = 0;int left = 0;int right = height.length - 1;while (left < right) {// 1. 先用当前的 left 和 right 计算面积int currentWidth = right - left;int currentHeight = Math.min(height[left], height[right]);int currentArea = currentWidth * currentHeight;// 2. 更新历史最大面积maxArea = Math.max(maxArea, currentArea);// 3. 然后再根据短板移动指针,为下一次循环做准备if (height[left] < height[right]) {left++;} else {right--;}}return maxArea;}
}

这样修改后,循环体内部的逻辑非常纯粹:对于当前的 left 和 right 状态,我们先计算它的面积并更新 maxArea,然后再决定下一步怎么走。每一步都处理一个独立的状态,非常清晰。

1.2 滑动窗口

滑动窗口模式,顾名思义,就像有一个大小可变的“窗口”在一个序列(通常是数组或字符串)上滑动。它也是用两个指针来定义的,通常我们叫它们 left 和 right,但这里的 left 和 right 都从起点开始,共同构成一个窗口 [left, right) 或 [left, right]

这个模式专门用来解决“连续子数组”或“连续子字符串”的相关问题,例如:

  • 找到满足某条件的最长/最短的连续子串。
  • 找到所有满足某条件的连续子数组的数量。

和左右指针定初始位置、判断条件往相反的方向走不同。
滑动窗口是两个指针相同一个方向移动:

  1. 窗口扩大: 不断移动 right 指针,让新的元素进入窗口。
  2. 更新状态: 每当新元素进入,就更新窗口内的信息(例如:窗口内元素的和、不同字符的数量等)。
  3. 判断收缩: 当窗口内的状态不再满足题目要求时(或者说,满足了收缩条件时),就需要移动 left 指针,将元素移出窗口,这个过程叫作窗口收缩
  4. 持续收缩: 持续移动 left 指针,直到窗口内的状态再次满足题目要求
  5. 重复以上过程,直到 right 指针到达序列末尾。

3. 无重复字符的最长子串

import java.util.HashSet;
import java.util.Set;class Solution {public int lengthOfLongestSubstring(String s) {int n = s.length();if (n == 0) {return 0;}// 1. 初始化Set<Character> charSet = new HashSet<>();int maxLen = 0;int left = 0; // 窗口左边界int right = 0; // 窗口右边界// 2. 循环,让 right 指针作为主导,探索整个字符串while (right < n) {// 3. 尝试扩大窗口:检查 right 指向的字符是否已在窗口中char charToadd = s.charAt(right);// 4. 如果有重复,触发窗口收缩//    注意这里是 while 循环,因为可能需要收缩多次while (charSet.contains(charToadd)) {// 从 Set 中移除 left 指向的字符charSet.remove(s.charAt(left));// 收缩窗口left++;}// 5. 此时窗口内一定没有重复了,可以安全地加入新字符charSet.add(charToadd);// 6. 更新最大长度//    每一次扩大窗口后,都计算一下当前长度,并尝试更新最大值maxLen = Math.max(maxLen, right - left + 1);// 7. 真正地扩大窗口,为下一次循环做准备right++;}return maxLen;}
}

还有209. 长度最小的子数组。

拓展:leetcode题解与题目汇总:powcai滑动窗口题解

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

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

相关文章

使用cookiecutter创建python项目

一、关于Python项目结构Python 项目并没有完全统一的 “固定结构”&#xff0c;但行业内有一些广泛遵循的约定俗成的目录结构&#xff08;尤其针对可分发的包或大型项目&#xff09;。同时&#xff0c;确实有工具可以快速生成这些标准化结构&#xff0c;提高开发效率&#xff0…

台积电生态工程深度解析:从晶圆厂到蜂巢的系统架构迁移

当半导体巨头将工厂视为生态系统&#xff0c;用工程思维解决环境问题概述&#xff1a;生态系统的工程化再造台积电近日开展的"积蜜"项目绝非简单的企业CSR行为&#xff0c;而是一场将生态系统视为复杂系统进行工程化改造的技术实践。本文将从系统架构、数据监控、循环…

从零实现一个简易计算器

最近在刷算法题时&#xff0c;遇到了实现计算器的问题。一开始觉得很简单&#xff0c;但真正动手实现时才发现其中有很多细节需要考虑。今天就来分享一下我的实现思路和学到的经验。问题分析我们需要实现一个能够处理加减乘除四则运算的计算器&#xff0c;要正确处理运算符的优…

Actix-webRust Web框架入门教程

文章目录引言Actix-web是什么&#xff1f;准备工作你的第一个Actix-web应用理解代码结构处理请求和响应接收请求数据返回响应中间件 - 增强你的应用状态管理和依赖注入实用示例&#xff1a;构建RESTful API测试你的Actix-web应用部署Actix-web应用结语额外资源引言 嘿&#xf…

若依框架前端通过 nginx docker 镜像本地运行

1. 前言 项目运行过程图&#xff1a;对于前端项目通过命令 npm run build 打包后&#xff0c;无法直接运行。存在如下错误&#xff1a;可以通过配置 nginx 服务器运行前端项目解决如上问题。 2. Nginx 运行 采用 docker 镜像的方式运行&#xff0c;docker-compose.yml 文件内容…

浅聊一下HTTP协议

在日常上网浏览网页、刷视频时&#xff0c;背后都离不开 HTTP 协议的支持。作为 Web 世界的 “交通规则”&#xff0c;它负责服务器和客户端浏览器之间的数据传输。这篇文章就带大家全面了解 HTTP 协议&#xff0c;从基本概念到通信细节&#xff0c;再到安全相关的 HTTPS&#…

机器人控制器开发(定位——cartographer ros2 使用2)

文章总览 1 纯定位模式 当完成建图后&#xff0c;会生成pbstream格式的地图文件 配置纯定位模式的lua脚本 backpack_2d_localization.lua include "backpack_2d.lua"TRAJECTORY_BUILDER.pure_localization_trimmer {max_submaps_to_keep 3, } POSE_GRAPH.optimi…

《大数据之路1》笔记3:数据管理

一 元数据 1.1 元数据概述 定义&#xff1a; 元数据是关于数据的数据&#xff0c;元数据打通了源数据、数据仓库、数据应用&#xff0c;记录了数据从生产到消费的全部过程。元数据主要记录数据仓库中模型的定义、各层级间的映射关系、监控数据仓库的数据状态和ETL的任务运行状态…

排序实现java

排序算法概述Java中实现排序可以通过多种方式&#xff0c;包括内置方法、自定义算法或使用第三方库。常见的排序算法有冒泡排序、选择排序、插入排序、快速排序、归并排序等。使用Arrays.sort()方法对于数组排序&#xff0c;Java提供了Arrays.sort()方法&#xff0c;支持对基本…

51c大模型~合集182

我自己的原文哦~ https://blog.51cto.com/whaosoft/14174587 #LaV-CoT 超越GPT-4o&#xff0c;蚂蚁集团与南洋理工大学提出&#xff1a;首个语言感知的视觉思维链 随着大型视觉语言模型&#xff08;VLM&#xff09;的飞速发展&#xff0c;它们在处理复杂的视…

C++ STL之deque的使用和模拟实现

目录 deque 核心本质与定位 与stack和queue的关系: deque的使用 deque的底层实现 deque的原理介绍 deque的缺陷 总结: deque deque文档 : deque 翻译: 双端队列 deque&#xff08;通常发音类似“deck”&#xff09;是“double-ended queue”&#xff08;双端队列&…

布草洗涤厂设备租赁押金原路退回系统—东方仙盟

设备租赁状态设备管理添加设备设备收押金设备退押金在布草洗涤行业的运营版图中&#xff0c;设备租赁是连接厂商与客户的重要纽带&#xff0c;而押金的收取与退还则是这一环节中关乎信任与效率的关键节点。未来之窗布草洗涤厂深谙此道&#xff0c;专为设备租赁业务打造的 “押金…

换源rocklinux和centos

一、Rockylinux换源&#xff0c;国外的源换成国内的源#nmcli connection modify ens33 ipv4.addresses 192.168.121.11 ipv4.gateway 192.168.121.2 ipv4.method manual ipv4.dns 114.114.114.114 connection.autoconnect yes修改地址#systemctl stop firewalld#systemctl diab…

第一部分:服务器硬件配置

目录1.1 服务器上架与连线1.2 启用CPU虚拟化功能&#xff08;BIOS设置&#xff09;1.3 配置RAID存储步骤1&#xff1a;进入RAID配置界面步骤2&#xff1a;确认RAID控制器信息步骤3&#xff1a;创建系统RAID&#xff08;用于安装ESXi&#xff09;步骤4&#xff1a;创建数据RAID&…

手搓一个 DELL EMC Unity存储系统健康检查清单

写在前面对于DELL EMC存储系统Unity的一些深度的健康检查通过Web的Unisphere图形化界面是做不到的&#xff0c;图形化界面只能看到是否有告警&#xff0c;物理的东西是否有问题的&#xff0c;逻辑的Pool和LUN等是否ready&#xff0c;再深入的潜在的问题是查不到的。另外&#x…

【数据结构】二叉树的概念

01 概念定义&#xff1a;二叉树既然叫二叉树&#xff0c;顾名思义即度最大为2的树称为二叉树。 它的度可以为 1 也可以为 0&#xff0c;但是度最大为 2 。 一颗二叉树是节点的一个有限集合&#xff0c;该集合&#xff1a;① 由一个根节点加上两棵被称为左子树和右子树的二叉树组…

【RK3576】【Android14】如何在Android14下单独编译kernel-6.1?

单独编译kernel依赖如下几个源码&#xff1a;【交叉编译工具链】prebuilts/clang/host/linux-x86/clang-r487747c【内核源码】kernel-6.1为什么Android下编译内核使用clang作为交叉编译工具链而不是GCC&#xff1f;Android 14 选择使用预置的 Clang 工具链&#xff08;如 clang…

什么是Redis的Pipeline

介绍Redis的Pipeline是一种网络优化技术&#xff0c;在没有Pipeline的时候&#xff0c;客户端往redis发送请求&#xff0c;客户端需要等到redis响应之后才能发送下一个请求。而Pipeline&#xff0c;使redis可以一次性接收多个请求。减少了通信次数&#xff0c;显著的提高了性能…

【ElementUI el-table跨页勾选】

一、el-table需加上refs和 row-key属性 二、type"selection"勾选框 需加上 reserve-selection储备选择属性 三、在分页请求数据时&#xff0c;触发 setSelected()方法 四、在 selection-change变化时保存 selectedRows <el-table ref"tables" :data&quo…

论文阅读/博弈论/拍卖:《Truthful Auction for Cooperative Communications》

摘要&#xff1a;一方面&#xff0c;协作通信由于其在提升无线网络容量方面的巨大潜力而日益受到关注。另一方面&#xff0c;协作通信技术的实际应用却很少见&#xff0c;即使在一些对带宽需求极高的应用场景中&#xff0c;系统设计者也并未采用协作通信技术来开发创新的网络解…