小白算法记录Day1| 704. 二分查找、27. 移除元素
一、LeetCode 704-二分查找
题目链接:704.二分查找
题目描述:给定一个n
个元素有序的(升序)整型数组 nums
和一个目标值 target
,写一个函数搜索nums
中的 target
,如果目标值存在返回下标,否则返回-1
。
输入: nums = [-1,0,3,5,9,12], target = 9
输出: 4
解释: 9 出现在 nums 中并且下标为 4
解题思路:
由于是升序数组,故本题使用二分法解答,定义左中右指针,从数组中间开始查找。
二分法-第一种写法(左闭右闭区间)
int search(int* nums, int numsSize, int target){
int left = 0;
int right = numsSize - 1;
int mid = 0;
while(left <= right) {
mid = (left + right) / 2;
if (nums[mid] == target)
return mid;
else if (nums[mid] < target)
left = mid + 1;
else if (nums[mid] > target)
right = mid - 1;
}
return -1;
}
二分法-第二种写法(左闭右开区间)
int search(int* nums, int numsSize, int target){
int left = 0;
int right = numsSize;
int mid = 0;
while(left < right) {
mid = (left + right) / 2;
if (nums[mid] == target)
return mid;
else if (nums[mid] < target)
left = mid + 1;
else if (nums[mid] > target)
right = mid;
}
return -1;
}
难点
本题有两个难点:
一、两种方法的while
循环内 <
和 <=
的区别;
二、当mid
指针所指向的值小于目标值时,下一个循环中right
所指向的位置。
分析如下:
1.right
的取值要根据区间的范围确定,如果是右闭的区间,right
指针定义时应该指向numsSize - 1
,此时数组范围内的所有数据都能被left、mid、right
取到;若是右开区间,循环为while(left < right)
此时right指针无法取到最右侧的数据(此时left = right
为非法);
2.当数组区间为左闭右闭时,即所有的数值都能取到,例如在数组[3,3]
中,此时left = right
在区间内是合法的、能取到的,故mid
指针所指向的值大于目标值时,right
指针要刷新为mid - 1
(之前的mid
对应的数值由于能取到,所以已经使用过了);
3.当数组区间为左闭右开时,即区间内所有的数值不是都能取到,例如定义right = numsSize
,实际上right
能取到的最大值为numsSize - 1
所对应的值,故mid
指针所指向的值大于目标值时,right
指针要刷新为mid1
,下一轮循环中能取到的最大值为mid1 - 1
;
4.当mid指针所指向的值小于目标值时,left
指针要刷新位置,更改为mid + 1
。
总结
其实很容易看出来解题方法,但是在处理过程中要注意上文中提到的两个点,如果能够看懂以上分析,代码还是很轻松能搞定的。
二、LeetCode 27-移除元素
题目链接:27.移除元素
题目描述:给你一个数组nums
nums 和一个值 val
,你需要 原地 移除所有数值等于val
的元素,并返回移除后数组的新长度。不要使用额外的数组空间,你必须仅使用 O(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],也会被视作正确答案。
解题思路:
做题前要清楚,移除数组中某一个的元素,并不是简单的将其删除就完了,而是将要移除的目标数值后的数将目标值覆盖,例如[1,2,3,4,5]
,移除过程为4
覆盖3
的位置,5
覆盖4
的位置,最初5
的位置放什么都无所谓了,一些编程语言会对更新后的数组进行包装,最初的数组size
会由5
变为4
,但并不是代表这个空间真正变成4
了,实际上物理空间还是5
,仅仅是最后一个元素没有处理而已。
双指针法:通过一个快指针和慢指针在一个for
循环下完成两个for
循环的工作。
定义快慢指针
快指针:寻找新数组的元素 ,新数组就是不含有目标元素的数组;
慢指针:指向更新新数组下标的位置。
移除元素-快慢指针法
int removeElement(int* nums, int numsSize, int val){
int slow = 0;
for(int fast = 0 ; fast < numsSize ; fast++) {
if(nums[fast] != val) {//当快指针当前指向的值不等于目标值时,将这个值赋给慢指针
nums[slow] = nums[fast];
slow++;
}
}
return slow;
}
总结
新手小白,第一次写博客,语言组织能力也不强,总感觉很难将思维转换成文字,排版能力也有待提高,花费了好多时间,希望将来能越来越好吧。