704. 二分查找
给定一个n个元素有序的(升序)整型数组nums 和一个目标值target ,写一个函数搜索nums中的 target,如果目标值存在返回下标,否则返回 -1。 示例 1: 输入: nums = [-1,0,3,5,9,12], target = 9 输出: 4 解释: 9 出现在 nums 中并且下标为 4 示例2: 输入: nums = [-1,0,3,5,9,12], target = 2 输出: -1 解释: 2 不存在 nums 中因此返回 -1
解题: 使用二分法的前提,有两个关键词——升序数组和无重复数值
在写代码的时候,有两个点需要注意,一个是区间的开闭,一个是middle给right和left的赋值
区间可以使用左闭右开和左闭右闭的区间,对应的middle赋值也不同,下面贴出左闭右闭的写法
做这道题的过程中先是纠结了除于2取整这个操作该如何优雅地去写,最后学习了随想录的解法,用了右移的操作符。另外在开闭区间的时候也思考有误,必须要注意
/**
* @param {number[]} nums
* @param {number} target
* @return {number}
*/
var search = function(nums, target) {
let left = 0
let right = nums.length - 1
let middle
while(left <= right) {
middle = left + ((right - left)>>1)
if (nums[middle] < target) {
left = middle + 1
} else if(nums[middle] > target) {
right = middle - 1
} else {
return middle
}
}
return -1
};
注:这里在获取middle的新值时,用了(right - left)>>1的方法,>>是二进制右移的一种运算方法,>>1相当于获取前一个值除于2的整数,<<1相当于前一个值乘于2的整数
27. 移除函数
给你一个数组 nums和一个值 val,你需要 原地 移除所有数值等于val的元素,并返回移除后数组的新长度。 不要使用额外的数组空间,你必须仅使用 O(1) 额外空间并 原地 修改输入数组。 元素的顺序可以改变。你不需要考虑数组中超出新长度后面的元素。
示例 1 输入:nums = [3,2,2,3], val = 3 输出:2, nums = [2,2] 解释:函数应该返回新的长度 2, 并且 nums 中的前两个元素均为 2。你不需要考虑数组中超出新长度后面的元素。例如,函数返回的新长度为 2 ,而 nums = [2,2,3,3] 或 nums = [2,2,0,0],也会被视作正确答案。 示例2: 输入:nums = [0,1,2,2,3,0,4,2], val = 2 输出:5, nums = [0,1,4,0,3] 解释:函数应该返回新的长度 5, 并且 nums 中的前五个元素为 0, 1, 3, 0, 4。注意这五个元素可为任意顺序。你不需要考虑数组中超出新长度后面的元素。
/**
* @param {number[]} nums
* @param {number} val
* @return {number}
*/
var removeElement = function(nums, val) {
let slow = 0
for(let fast = 0;fast < nums.length;fast++) {
if (nums[fast] !== val) {
nums[slow] = nums[fast]
slow++
}
}
return slow
};
解题: 数组在内存中的存储方式是连续的,删除操作实际上是使用新值来覆盖旧值,而删除操作并不会影响到数组实际的存储长度。
这道题的思路是运用了双指针,通过快指针的遍历,将不等于val的值直接赋值给慢指针的位置,双指针的思路非常巧妙!