今天的题目主要都是力扣前100中,关于双指针的题
1.移动零
链接:移动零
示例:
示例 :
输入: nums =[0,1,0,3,12]
输出:[1,3,12,0,0]
可以看到保持原有元素的顺序,将所有的0,移动到数组最后方即可。
这道题看见思路很明确,就是遍历数组,碰到0了,进行位置互换。这个题目实际上和数组的元素移除几乎是一样的,只不过元素删除是进行覆盖,而移动0是进行元素移动
思路:
(1)定义两个指针,一个为快指针(负责遍历数组),另一个为慢指针(负责标记0)
(2)快指针进行移动遍历,如果发现0元素,则不进行任何操作,如果不是0元素,则进行与慢指针的元素互换
代码如下:
public class MobileZero {
/*
移动0
*/
//思路:双指针法
//快慢指针同时移动,一个指针用于标记0,一个指针移动,非0则交换两个指针,0则继续移动
public void moveZeroes(int[] nums) {
//慢指针:标记非0元素的位置
int slowCur = 0;
//快指针:遍历数组
for (int cur = 0; cur < nums.length; cur++) {
if (nums[cur] != 0){
int temp = nums[cur];
nums[cur] = nums[slowCur];
nums[slowCur] = temp;
//移动慢指针
slowCur++;
}
}
}
}
2.三数之和
链接:三数之和
示例:
给你一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 a,b,c ,使得 a + b + c = 0 ?请你找出所有满足条件且不重复的三元组。
给定数组 nums = [-1, 0, 1, 2, -1, -4],
满足要求的三元组集合为: [ [-1, 0, 1], [-1, -1, 2] ]
这道题我的大概思路就是,遍历所有元素,并找到每个元素对应的两个符合条件的数字,那既然需要两个数字,就需要使用双指针了
那么下一个问题就是,指针该指向哪里,并如何移动呢?
首先就需要将数组进行排序,因为有了顺序性,才有移动的规律,当我们将数组排序后
[-4,-1,-1,0,1,2]
遍历整个数组的所有元素,并把其当作a
三种情况:
1.a+b+c=0 这种情况符合要求,添加到列表中
2.a+b+c>0 这种情况说明,a+b<c 所以c要减小
3.a+b+c<0 这种情况说明,a+b>c 所以b要增大
换成指针来说,减小和增大就要移动,那么位置我设置的头指针指向a+1,尾指针指向数组末尾
第二种情况:尾指针向前移动
第三种情况:头指针向后移动
代码如下:
import java.util.*;
public class SumOfThreeNumbers {
/*
三数之和
*/
public List<List<Integer>> threeSum(int[] nums) {
//思路:a + b + c = 0
//采用双指针思路,遍历所有元素,并把其当作a,然后寻找 b + c = -a 的元素
//头指针指向a+1,尾指针指向数组末尾,数组排序
//遍历整个数组的所有元素,寻找每个元素的对应数值
//三种情况:
//1.a+b+c=0 这种情况符合要求,添加到列表中
//2.a+b+c>0 这种情况说明,a+b<c 所以c要减小,尾指针向前移动
//3.a+b+c<0 这种情况说明,a+b>c 所以b要增大,头指针向后移动
//去重处理:在进入遍历数组时,判断当前元素是否等于上一个元素,如果相等则跳过该元素的遍历
//对指针进行处理:因为指针的遍历也会重复,所以采用while循环
// 如果移动的下一个元素相等,那么指向下一个元素,然后进行自增
List<List<Integer>> result = new ArrayList<>();
Arrays.sort(nums);
for (int i = 0; i < nums.length; i++) {
//去重操作,防止遍历的元素重复
if (i > 0 && nums[i] == nums[i - 1]){
continue;
}
//在每次循环中,来移动头指针和尾指针
int left = i + 1;
int right = nums.length - 1;
//移动指针并寻找满足条件的元素
while(left < right){
int num = nums[i] + nums[left] + nums[right];
//满足条件
if (num == 0){
List<Integer> list = new ArrayList<>();
list.add(nums[i]);
list.add(nums[left]);
list.add(nums[right]);
result.add(list);
//去重处理:防止指针所指向的元素重复
while(left < right && nums[left] == nums[left + 1]){
left++;
}
while(left < right && nums[right] == nums[right - 1]){
right--;
}
//移动指针,寻找当前元素的其他对应元素
left++;
right--;
} else if (num > 0) {
//c过大,尾指针左移
right--;
}else {
//b过小,头指针右移
left++;
}
}
}
return result;
}
}
3.盛最多水的容器
链接:盛最多水的容器
给定一个长度为 n
的整数数组 height
。有 n
条垂线,第 i
条线的两个端点是 (i, 0)
和 (i, height[i])
。
找出其中的两条线,使得它们与 x
轴共同构成的容器可以容纳最多的水。
返回容器可以储存的最大水量。
示例 :
输入:[1,8,6,2,5,4,8,3,7]
输出:49
解释:图中垂直线代表输入数组 [1,8,6,2,5,4,8,3,7]。在此情况下,容器能够容纳水(表示为蓝色部分)的最大值为 49。
思路:
简单理解,就是寻找最大乘积,底 x 高的最大值,其实这里可以直接使用循环不去思考,双指针移动,遍历每一根柱子与其他柱子形成的底,和取两个柱子最短的进行相乘,找到最大值就好了,但是会超时,所以,又要考虑双指针的移动问题(选高柱子会漏水)
首先将问题拆分,最大值无非就是找到,底最大,高最大的值,如果我们先从高入手,那么就是将数组进行排序,可是这样,毫无规律,又要进行遍历,所以试试从底入手
那么最大底,一定是第一个柱子和最后一个柱子形成的底最大,然后比较两者谁更矮,移动指向矮的那一边的指针
因为下一个指向,如果还比最后一个柱子矮,则会继续向尾部移动寻找,如果高,尾部向头部移动。
代码如下:
import java.util.HashMap;
public class TheContainerWithTheMostWaterCapacity {
/*
盛水最多的容器
*/
public int maxArea(int[] height) {
//思路:定义两个指针,一个指向第一个值,一个指向最后一个值
//根据索引计算底宽,根据值的大小判断高,小的做高
//哪边的值小,则对应的指针移动,更新最大值
int beforeCur = 0;
int behindCur = height.length - 1;
int maxVolume = 0;
while (beforeCur < behindCur) {
int bottom = behindCur - beforeCur;
int high = Math.min(height[beforeCur], height[behindCur]);
maxVolume = Math.max(maxVolume, bottom * high);
if (height[beforeCur] > height[behindCur]) {
behindCur--;
} else {
beforeCur++;
}
}
return maxVolume;
}
}
今天的题更多的是思路吧,比起前面的基础数据结构知识,会更复杂,更考验读题,今天的练习就到这里!谢谢大家