【无标题】

小白算法记录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;
}

总结

新手小白,第一次写博客,语言组织能力也不强,总感觉很难将思维转换成文字,排版能力也有待提高,花费了好多时间,希望将来能越来越好吧。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值