二分搜索、移除元素、有序数组的平方、长度最小的子数组

力扣704. 二分搜索

其思想是在有序数组中,每次通过比较中间元素,把查找区间缩小一半,从而快速定位目标值。

先从数组的长度设定left和right,之后再循环判断middle所在位置的数与target的大小,从而查找目标

java 复制代码
class Solution {
    public int search(int[] nums, int target) {
        int left=0;
        int right=nums.length-1;
        while(left<=right){
            int middle=(left+right)/2;
            if(nums[middle]>target){
                right=middle-1;
            }
            else if(nums[middle]<target){
                left=middle+1;
            }
            else
            return middle;
        }
        return -1;
    }
}

1.要确定是左闭右闭还是左开右闭或者坐闭右开

2.如上代码是左闭右闭,while中left可以等于right,举个例子:[left,right]若都为1,[1,1]这是可以的。

3.若是左闭右开区间,则left不能等于right,且最开始初始化right时,不必减一,且在判断(nums[middle]>target时,right=middle,不再减一。

java 复制代码
class Solution {
    public int search(int[] nums, int target) {
        int left = 0;
        int right = nums.length ;
        while (left < right) {
            int middle = left + (right - left) / 2;
            if (nums[middle] > target) {
                right = middle ;
            } else if (nums[middle] < target) {
                left = middle + 1;
            } else {
                return middle;
            }
        }
        return -1;
    }
}

力扣27.移除元素

由于数组是连续的,所以如果要移除中间的一个数,那么后面的就需要向前覆盖。最简单的是嵌套循环来写。

第一个循环用来遍历数组,如果是要删除的数,则再嵌套一个循环,将后面的数向前移。代码如下

java 复制代码
class Solution {
    public int removeElement(int[] nums, int val) {
        int l=nums.length;
        for(int i=0;i<l;i++){
            if(nums[i]==val){
                for(int j=i;j<l-1;j++){
                    nums[j]=nums[j+1];
                }
                l--;
                i--;
            }
        }
        return l;
    }
}

如果想要只使用一个for循环,则可以使用双指针的办法。初始化两个指针,全部指向数组下标为0的地方,fast向前走,如果不是val,则slow将fast的值写入数组;如果是val则不写入数组。当然代码如下

java 复制代码
class Solution {
    public int removeElement(int[] nums, int val) {
       int fast=0;
       int slow=0;
       for(int i=0;i<nums.length;i++){
        if(nums[fast]!=val){
            nums[slow++]=nums[fast++];
        }
        else{
            fast++;
        }
       }
       return slow;
    }
}

977.有序数组的平方

这题需要把数组中的数平方,并按照由小到大的顺序排放,数组一开始就是由小到大,但里面会有负数,所以我们可以直接暴力求解,全部平方再使用快排进行排序。复杂度是O(nlogn).

当然也有别的办法,也是双指针。在数组开头和末尾设置两个指针,平方的话这两个其中一个是最大的,我们再创建一个新数组,倒着把平方后的数放进去。代码如下

java 复制代码
class Solution {
    public int[] sortedSquares(int[] nums) {
        int k=nums.length-1;
        int[] result=new int[nums.length];
        for(int i=0,j=nums.length-1;i<=j;){
            if(nums[i]*nums[i]>nums[j]*nums[j]){
                result[k--]=nums[i]*nums[i];
                i++;
            }else
            {
                result[k--]=nums[j]*nums[j];
                j--;
            }
        }
        return result;
    }
}

209.长度最小的子数组

题目要求总和大于等于target的长度最小的**子数组,**可以使用暴力的方法,两个循环嵌套一个一个找。还可以使用滑动窗口的方法。

这里就只讲滑动窗口,这里借用卡尔的图(我也是看的他的),要有两个指针,i指向起,j指向终。先将j往后走,并将数组中的每个数加起来,等到大于等于target时,进入while循环,将此时的长度赋给subl,再用min找出最小的长度赋给result,这个长度就是j-i+1,但此时不一定是最小的,我们把nums[i]减掉,有可能还是>=target,i++,此时接着循环,找出最小长度,以此类推。直到sum<target,出while循环,j++,再循环下一轮。最后返回result,当然也会出现所有数都加完了,但依然小于target,此时我们返回0,这里我用 if(result==nums.length+1)来判断。

java 复制代码
class Solution {
    public int minSubArrayLen(int target, int[] nums) {
        int i=0;
        int sum=0;
        int result=nums.length+1;
        int subl=0;
        for(int j=0;j<nums.length;j++){
            sum+=nums[j];
            while(sum>=target){
                subl=j-i+1;
                result=Math.min(result,subl);
                sum=sum-nums[i];
                i++;
            }
        }
        if(result==nums.length+1)
        return 0;
        return result;
    }
}
相关推荐
Dlrb12119 小时前
C语言-指针三
c语言·算法·指针·const·命令行参数
Tisfy9 小时前
LeetCode 2540.最小公共值:双指针(O(m+n))
算法·leetcode·题解·双指针
一直不明飞行9 小时前
Java的equals(),hashCode()应该在什么时候重写
java·开发语言·jvm
IronMurphy9 小时前
【算法四十七】152. 乘积最大子数组
算法
REDcker9 小时前
有限状态机与状态模式详解 FSM建模Java状态模式与C++表驱动模板实践
java·c++·状态模式
你的保护色10 小时前
【无标题】
java·服务器·网络
basketball61610 小时前
C++ 构造函数完全指南:从入门到进阶
java·开发语言·c++
淘矿人11 小时前
Claude辅助DevOps实践
java·大数据·运维·人工智能·算法·bug·devops
Cosolar11 小时前
万字详解:RAG 向量索引算法与向量数据库架构及实战
数据库·人工智能·算法·数据库架构·milvus
小江的记录本11 小时前
【Java基础】泛型:泛型擦除、通配符、上下界限定(附《思维导图》+《面试高频考点清单》)
java·数据结构·后端·mysql·spring·面试·职场和发展