一起养成写作习惯!这是我参与「掘金日新计划 · 4 月更文挑战」的第8天,点击查看活动详情。
Hi, 大家好。我是新人,程序员库里。
今后会按类分享算法题。
今天给大家分享第8道leetcode上数组中使用双指针相关的算法题。
167. 两数之和 II - 输入有序数组
leetcode地址:leetcode-cn.com/problems/tw…
难度
中等
描述
给你一个下标从 1 开始的整数数组 numbers ,该数组已按 非递减顺序排列 ,请你从数组中找出满足相加之和等于目标数 target 的两个数。如果设这两个数分别是 numbers[index1] 和 numbers[index2] ,则 1 <= index1 < index2 <= numbers.length 。
以长度为 2 的整数数组 [index1, index2] 的形式返回这两个整数的下标 index1 和 index2。
你可以假设每个输入 只对应唯一的答案 ,而且你 不可以 重复使用相同的元素。
你所设计的解决方案必须只使用常量级的额外空间。
示例1
输入:numbers = [2,7,11,15], target = 9
输出:[1,2]
解释:2 与 7 之和等于目标数 9 。因此 index1 = 1, index2 = 2 。返回 [1, 2] 。
示例2
输入:numbers = [2,3,4], target = 6
输出:[1,3]
解释:2 与 4 之和等于目标数 6 。因此 index1 = 1, index2 = 3 。返回 [1, 3] 。
示例3
输入:numbers = [-1,0], target = -1
输出:[1,2]
解释:-1 与 0 之和等于目标数 -1 。因此 index1 = 1, index2 = 2 。返回 [1, 2] 。
原理
- 采用对撞指针的方法,利用数组是升序的。
- 定义一个变量left,初始值是0,指向数组第一个位置,表示左指针
- 定义一个变量right,初始值是 numbers.length-1,指向数组最后一个位置,表示右指针
- 开始进行循环,条件是当left小于right的时候
- 当 左指针指向的元素加上右指针指向的元素刚好等于目标值, 此时表明找到了这两个元素,因为题目要求是索引从1开始,所以返回[left+1, right+1]
- 当 左指针指向的元素加上右指针指向的元素 小于 目标值,因为数组是升序的,所以需要将左指针向右移动一位,才有可能使得左指针指向的元素加上右指针指向的元素 等于 目标值,即 left++
- 当 左指针指向的元素加上右指针指向的元素 大于 目标值,因为数组是升序的,所以需要将右指针向左移动一位,才有可能使得左指针指向的元素加上右指针指向的元素 等于 目标值,即 right--
- 当遍历完之后,因为题目要求是有解的,所以最后返回 [left+1, right+1]
代码
/**
* @param {number[]} numbers
* @param {number} target
* @return {number[]}
*/
var twoSum = function(numbers, target) {
let l = 0 , r = numbers.length - 1;
while(l< r){
if(numbers[l] + numbers[r] === target){
return [l+1,r+1]
}else if(numbers[l] + numbers[r] < target){
l++
}else{
r--
}
}
};