【LeetCode Hot100 双指针】移动零、盛最多水的容器、三数之和、接雨水
双指针
- 1. 移动零
- 题目描述
- 解题思路
- 关键思路:
- 步骤:
- 时间复杂度:
- 空间复杂度:
- 代码实现
- 2. 盛最多水的容器
- 题目解析
- 解题思路
- 代码实现
- 3. 三数之和
- 问题描述:
- 解题思路:
- 算法步骤:
- 代码实现:
- 4. 接雨水
- 问题描述:
- 解题思路:
- 算法步骤:
- 代码实现:
1. 移动零
题目描述
给定一个数组 nums,编写一个方法,将所有零元素移动到数组的末尾,同时保持非零元素的相对顺序不变。必须 原地 修改数组,且不能使用额外的数组。
解题思路
此问题要求将数组中的所有零元素移动到末尾,保持非零元素的相对顺序不变,并且必须在原数组上修改。常规的做法可能会用两个临时数组,但这里需要在 O(1) 空间复杂度下完成任务。因此,我们需要设计一个在原数组上操作的解法。
关键思路:
我们可以使用 双指针法 来解决该问题:
- 指针
p: 用于遍历数组中的元素,用来指向需要替换的位置(即当前非零元素应该放的位置)。 - 指针
q: 用于遍历数组中的元素,指向当前正在检查的元素。
步骤:
-
初始化两个指针
p和q:p指向当前可以插入非零元素的位置。q用来遍历整个数组,检查每一个元素。
-
遍历数组:
- 当
nums[p]为零时:- 如果
nums[q]也为零,直接跳过q,继续检查下一个元素。 - 如果
nums[q]为非零元素,交换nums[p]和nums[q],将非零元素移到前面,同时将p和q向后移动。
- 如果
- 当
nums[p]不是零时,直接将p向后移动,继续检查下一个位置。
- 当
-
结束条件: 当
q遍历完数组时,操作结束。
时间复杂度:
- 单遍历数组,时间复杂度为 O(n),其中
n是数组的长度。
空间复杂度:
- 使用了常数空间,仅使用了两个指针,空间复杂度为 O(1)。
代码实现
class Solution {public void moveZeroes(int[] nums) {int p = 0, q = 1;while (q < nums.length) {if (nums[p] == 0) {while (q < nums.length - 1 && nums[q] == 0) {q++;}nums[p] = nums[q];nums[q] = 0;} p++;q++;}}
}
2. 盛最多水的容器
题目解析
题目要求我们在一个给定的整数数组 height[] 中,找到两条垂直线所构成的容器的最大面积。两条线的 x 轴坐标为数组中的两个元素位置,而容器的高度则是由这两个位置的较短的线决定的。
我们可以利用 双指针法 来解决这个问题,这样可以在一次遍历中找到最大面积,且时间复杂度为 O(n)。
解题思路
-
定义两个指针:
- 初始化两个指针
p和q,分别指向数组的两端。即p = 0,q = height.length - 1。
- 初始化两个指针
-
计算面积:
- 在每一步中,我们计算当前两条垂直线之间的面积,面积的计算公式为:
area = ( q − p ) × min ( height[p] , height[q] ) \text{area} = (\text{q} - \text{p}) \times \min(\text{height[p]}, \text{height[q]}) area=(q−p)×min(height[p],height[q]) q - p是两条线之间的宽度,min(height[p], height[q])是这两条线所能组成的容器的高度。
- 在每一步中,我们计算当前两条垂直线之间的面积,面积的计算公式为:
-
移动指针:
- 为了找到可能的最大面积,我们需要移动其中较短的线来尝试提高容器的高度。我们移动短板的指针,若
height[p] < height[q],则p++;否则,q--。通过这种方式,我们不断调整指针,寻找可能的最大面积。
- 为了找到可能的最大面积,我们需要移动其中较短的线来尝试提高容器的高度。我们移动短板的指针,若
-
结束条件:
- 当
p与q相遇时,所有可能的组合都已经检查过,我们返回最大面积。
- 当
-
为什么移动短板
- 面积是由短板来决定的,如果选择移动长板保留短板,那么移动后的面积不可能比移动前的更大了。 假设移动长板后得到的新板长度比之前的短板长,那么容器的高度仍然是短板的长度,而宽度减小了;如果移动长板后得到的新板长度比之前的短板短,那么面积就更小了。
代码实现
class Solution {public int maxArea(int[] height) {int p = 0, q = height.length - 1; // 初始化指针int result = 0; // 用来存储最大面积while (p < q) {int area = (q - p) * Math.min(height[p], height[q]); // 计算当前的面积result = Math.max(result, area); // 更新最大面积// 移动短板if (height[p] < height[q]) {p++; // 如果左边的高度小,移动左指针} else {q--; // 如果右边的高度小,移动右指针}}return result; // 返回最大面积}
}
3. 三数之和
问题描述:
给定一个整数数组 nums,找出所有和为零的三元组,要求不重复地返回结果。
解题思路:
这个问题要求在数组中找到所有和为零的三元组,并且不允许返回重复的三元组。考虑到以下几点:
- 排序:首先,我们可以将数组
nums进行排序,这样可以更方便地处理重复元素并使用双指针方法进行优化。 - 去重:由于数组可能包含重复的元素,我们需要跳过重复的三元组。可以通过跳过相同的元素来避免重复结果。
- 双指针:对于每个固定的
i,我们使用双指针j和k来遍历剩余的数组,寻找和为零的元素。这样可以有效地减少时间复杂度。 - 优化:通过判断最小的三个数和最大的两个数的和的情况,提前排除无解的情况,减少不必要的计算。
算法步骤:
- 排序数组:对
nums数组进行排序。 - 遍历数组:用一个
i来遍历数组,确保nums[i]是固定的。 - 使用双指针:对于每一个固定的
i,使用两个指针j和k来遍历数组中的元素,查找nums[i] + nums[j] + nums[k] == 0的三元组。 - 跳过重复元素:跳过重复的元素,以避免返回重复的三元组。
- 优化判断:通过判断数组中最小和最大的元素的和来减少不必要的计算。
代码实现:
class Solution {public List<List<Integer>> threeSum(int[] nums) {// 先把数组变成有序的数组Arrays.sort(nums);List<List<Integer>> ans = new ArrayList<>();int n = nums.length;// 最后留两个位置给 j 和 kfor (int i = 0; i < n - 2; i++) {if (i > 0 && nums[i] == nums[i - 1]) {// 如果和上一个数值相同就跳过,避免重复continue;}int j = i + 1;int k = n - 1;// 优化 1: 最小的三个数加和 > 0 就说明没有 j, k 加一起能 = 0,都 > 0if (nums[i] + nums[i + 1] + nums[i + 2] > 0) break;// 优化 2:nums[i] 和两个最大的数相加都 < 0,说明当前 i 和后面的任意 j, k 相加都 < 0if (nums[i] + nums[n - 1] + nums[n - 2] < 0) continue;while (j < k) {int s = nums[i] + nums[j] + nums[k];if (s == 0) {List<Integer> list = new ArrayList<>();list.add(nums[i]);list.add(nums[j]);list.add(nums[k]);ans.add(list);j++;// 避免重复while (j < k && nums[j] == nums[j - 1]) {j++;}k--;// 避免重复while (k > j && nums[k] == nums[k + 1]) {k--;}} else if (s > 0) {k--;} else {j++;}}}return ans;}
}
4. 接雨水
问题描述:
给定一个整数数组 height,其中每个元素代表一个柱子的高度。请计算能够接住多少个雨水。
解题思路:
此问题是经典的“接雨水”问题。我们需要计算每个位置能接住的雨水量。对于每个位置,能接住的雨水量取决于该位置的左边和右边的最大高度。因此,我们可以将问题转化为以下步骤:
- 计算每个位置的左边最大高度:对于数组中的每个位置
i,我们可以记录从左到右遍历时,当前位置左边的最大高度。这样可以确定该位置上方能够容纳多少水。 - 计算每个位置的右边最大高度:对于数组中的每个位置
i,我们可以记录从右到左遍历时,当前位置右边的最大高度。同样,这也帮助我们计算该位置上方能容纳多少水。 - 计算每个位置的水量:每个位置的水量由当前高度、左边最大高度和右边最大高度共同决定。具体来说,当前位置能容纳的水量为:
Math.min(left_max, right_max) - height[i]。
算法步骤:
- 创建两个数组
pre_max和suf_max:分别记录每个位置的左边最大高度和右边最大高度。 - 计算左边最大高度:从左到右遍历,更新
pre_max[i]为当前位置i及其左边所有位置的最大高度。 - 计算右边最大高度:从右到左遍历,更新
suf_max[i]为当前位置i及其右边所有位置的最大高度。 - 计算接住的雨水:对于每个位置
i,水量为Math.min(pre_max[i], suf_max[i]) - height[i],然后将水量累加得到总水量。
代码实现:
class Solution {public int trap(int[] height) {int ans = 0;int n = height.length;int[] pre_max = new int[n];int[] suf_max = new int[n];// 计算左边最大高度pre_max[0] = height[0];for (int i = 1; i < n; i++) {pre_max[i] = Math.max(pre_max[i-1], height[i]);}// 计算右边最大高度suf_max[n-1] = height[n-1];for (int i = n-2; i >= 0; i--) {suf_max[i] = Math.max(suf_max[i+1], height[i]);}// 计算每个位置的水量for (int i = 0; i < n; i++) {ans += Math.min(pre_max[i], suf_max[i]) - height[i];}return ans;}
}
相关文章:
【LeetCode Hot100 双指针】移动零、盛最多水的容器、三数之和、接雨水
双指针 1. 移动零题目描述解题思路关键思路:步骤:时间复杂度:空间复杂度: 代码实现 2. 盛最多水的容器题目解析解题思路代码实现 3. 三数之和问题描述:解题思路:算法步骤:代码实现: …...
HTML应用指南:利用POST请求获取接入比亚迪业态的充电桩位置信息
在新能源汽车快速发展的今天,充电桩的分布和可用性成为了影响用户体验的关键因素之一。比亚迪作为全球领先的新能源汽车制造商,不仅在车辆制造方面取得了卓越成就,也在充电基础设施建设上投入了大量资源。为了帮助用户更方便地找到比亚迪充电桩的位置,本篇文章,我们将探究…...
Android车机DIY开发之软件篇(十二) AOSP12下载编译
Android车机DIY开发之软件篇(十二) AOSP12下载编译 sudo apt-get update sudo apt-get install git-core gnupg flex bison gperf build-essential zip curl zlib1g-dev gcc-multilib gmultilib libc6-dev-i386 lib32ncurses5-dev libx11-dev lib32z-dev ccache libgl1-mesa-…...
Jenkins+gitee 搭建自动化部署
Jenkinsgitee 搭建自动化部署 环境说明: 软件版本备注CentOS8.5.2111JDK1.8.0_211Maven3.8.8git2.27.0Jenkins2.319最好选稳定版本,不然安装插件有点麻烦 一、安装Jenkins程序 1、到官网下载相应的版本war或者直接使用yum安装 Jenkins官网下载 直接…...
【文本处理】如何在批量WORD和txt文本提取手机号码,固话号码,提取邮箱,删除中文,删除英文,提取车牌号等等一些文本提取固定格式的操作,基于WPF的解决方案
企业的应用场景 数据清洗:在进行数据导入或分析之前,往往需要对大量文本数据进行预处理,比如去除文本中的无关字符(中文、英文),只保留需要的联系信息(手机号码、固话号码、邮箱)。…...
Linux系统引导与服务管理
目录 一、Linux引导过程 1、引导过程概述 1.1、BIOS开机自检 1.2、MBR读取 1.3、加载引导加载程序(GRUB) 1.4、内核加载 1.5、初始化进程(init) 二、服务 2.1、服务类型 2.2、服务管理工具 三、运行级别 四、systemd …...
网络工程师 (30)以太网技术
一、起源与发展 以太网技术起源于20世纪70年代,最初由Xerox公司的帕洛阿尔托研究中心(PARC)开发。最初的以太网采用同轴电缆作为传输介质,数据传输速率为2.94Mbps(后发展为10Mbps),主要用于解决…...
react项目引入tailwindcss不生效解决方案
根据tailwindcss官网的操作步骤下来,样式未生效,且未报错,看了挺多的资料,还是并未解决。 后面在另一个项目尝试时,报了下面的问题: Error: PostCSS plugin tailwindcss requires PostCSS 8 根据这个链接…...
【C#】条件运算符
1.逻辑与(&&) Console.WriteLine(true && true);//true Console.WriteLine(true && false);//false Console.WriteLine(false && false);//false2.逻辑或(||) Console.WriteLine(true || true);//true Console.WriteLine(true || false);//t…...
Windows11+PyCharm利用MMSegmentation训练自己的数据集保姆级教程
系统版本:Windows 11 依赖环境:Anaconda3 运行软件:PyCharm 一.环境配置 通过Anaconda Prompt(anaconda)打开终端创建一个虚拟环境 conda create --name mmseg python3.93.激活虚拟环境 conda activate mmseg 4.安装pytorch和cuda tor…...
WPS计算机二级•文档的文本样式与编号
听说这是目录哦 标题级别❤️新建文本样式 快速套用格式🩷设置标题样式 自定义设置多级编号🧡使用自动编号💛取消自动编号💚设置 页面边框💙添加水印🩵排版技巧怎么分栏💜添加空白下划线&#x…...
Word中Ctrl+V粘贴报错问题
Word中CtrlV粘贴时显示“文件未找到:MathPage.WLL”的问题 Word的功能栏中有MathType,但无法使用,显示灰色。 解决方法如下: 首先找到MathType安装目录下MathPage.wll文件以及MathType Commands 2016.dotm文件,分别复…...
python-leetcode 24.回文链表
题目: 给定单链表的头节点head,判断该链表是否为回文链表,如果是,返回True,否则,返回False 输入:head[1,2,2,1] 输出:true 方法一:将值复制到数组中后用双指针法 有两种常用的列表实现&#…...
数据治理双证通关经验分享 | CDGA/CDGP备考全指南
历经1个月多的系统准备,本人于2024年顺利通过DAMA China的CDGA(数据治理工程师)和CDGP(数据治理专家)双认证。现将备考经验与资源体系化整理,助力从业者高效通关。 🌟 认证价值与政策背景 根据…...
3.4 学习UVM中的uvm_monitor类分为几步?
文章目录 前言1. 定义2. 核心功能3. 适用场景4. 使用方法5. 完整代码示例5.1 事务类定义5.2 Monitor 类定义5.3 Scoreboard 类定义5.4 测试平台 6. 代码说明7. 总结 前言 以下是关于 UVM 中 uvm_monitor 的详细解释、核心功能、适用场景、使用方法以及一个完整的代码示例&…...
Java在大数据处理中的应用:从MapReduce到Spark
Java在大数据处理中的应用:从MapReduce到Spark 大数据时代的到来让数据的存储、处理和分析变得前所未有的重要。随着数据量的剧增,传统的单机计算方式已经无法满足处理需求。为了解决这个问题,许多分布式计算框架应运而生,其中Ma…...
日常吐槽。
一、写在前面 stereopy日常出bug(github issue里得有一半的问题是我提的,当然也有可能是因为我菜),stereopy自己生成的anndata自己不能计算空间共现关系,还是靠squidpy才能计算。另外还要一些函数一开并行计算就报错,这里留一些s…...
2025最新版Node.js下载安装~保姆级教程
1. node中文官网地址:http://nodejs.cn/download/ 2.打开node官网下载压缩包: 根据操作系统不同选择不同版本(win7系统建议安装v12.x) 我这里选择最新版win 64位 3.安装node ①点击对话框中的“Next”,勾选同意后点…...
机器学习:学习记录(二)
1. 机器学习中的常用函数 logistic函数(sigmoid函数):非线性激活函数,将R区间映射到(0,1)区间 ReLU函数:非线性激活函数,简单可以写作max(0,x),在0处不可导,但是可以人为定义其导数…...
迁移学习 Transfer Learning
迁移学习(Transfer Learning)是什么? 迁移学习是一种机器学习方法,它的核心思想是利用已有模型的知识来帮助新的任务或数据集进行学习,从而减少训练数据的需求、加快训练速度,并提升模型性能。 …...
【网络】每天掌握一个Linux命令 - iftop
在Linux系统中,iftop是网络管理的得力助手,能实时监控网络流量、连接情况等,帮助排查网络异常。接下来从多方面详细介绍它。 目录 【网络】每天掌握一个Linux命令 - iftop工具概述安装方式核心功能基础用法进阶操作实战案例面试题场景生产场景…...
零门槛NAS搭建:WinNAS如何让普通电脑秒变私有云?
一、核心优势:专为Windows用户设计的极简NAS WinNAS由深圳耘想存储科技开发,是一款收费低廉但功能全面的Windows NAS工具,主打“无学习成本部署” 。与其他NAS软件相比,其优势在于: 无需硬件改造:将任意W…...
家政维修平台实战20:权限设计
目录 1 获取工人信息2 搭建工人入口3 权限判断总结 目前我们已经搭建好了基础的用户体系,主要是分成几个表,用户表我们是记录用户的基础信息,包括手机、昵称、头像。而工人和员工各有各的表。那么就有一个问题,不同的角色…...
ESP32 I2S音频总线学习笔记(四): INMP441采集音频并实时播放
简介 前面两期文章我们介绍了I2S的读取和写入,一个是通过INMP441麦克风模块采集音频,一个是通过PCM5102A模块播放音频,那如果我们将两者结合起来,将麦克风采集到的音频通过PCM5102A播放,是不是就可以做一个扩音器了呢…...
12.找到字符串中所有字母异位词
🧠 题目解析 题目描述: 给定两个字符串 s 和 p,找出 s 中所有 p 的字母异位词的起始索引。 返回的答案以数组形式表示。 字母异位词定义: 若两个字符串包含的字符种类和出现次数完全相同,顺序无所谓,则互为…...
[Java恶补day16] 238.除自身以外数组的乘积
给你一个整数数组 nums,返回 数组 answer ,其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法,且在 O(n) 时间复杂度…...
rnn判断string中第一次出现a的下标
# coding:utf8 import torch import torch.nn as nn import numpy as np import random import json""" 基于pytorch的网络编写 实现一个RNN网络完成多分类任务 判断字符 a 第一次出现在字符串中的位置 """class TorchModel(nn.Module):def __in…...
10-Oracle 23 ai Vector Search 概述和参数
一、Oracle AI Vector Search 概述 企业和个人都在尝试各种AI,使用客户端或是内部自己搭建集成大模型的终端,加速与大型语言模型(LLM)的结合,同时使用检索增强生成(Retrieval Augmented Generation &#…...
高效线程安全的单例模式:Python 中的懒加载与自定义初始化参数
高效线程安全的单例模式:Python 中的懒加载与自定义初始化参数 在软件开发中,单例模式(Singleton Pattern)是一种常见的设计模式,确保一个类仅有一个实例,并提供一个全局访问点。在多线程环境下,实现单例模式时需要注意线程安全问题,以防止多个线程同时创建实例,导致…...
MySQL 8.0 事务全面讲解
以下是一个结合两次回答的 MySQL 8.0 事务全面讲解,涵盖了事务的核心概念、操作示例、失败回滚、隔离级别、事务性 DDL 和 XA 事务等内容,并修正了查看隔离级别的命令。 MySQL 8.0 事务全面讲解 一、事务的核心概念(ACID) 事务是…...
