算法第一天

704 二分查找

力扣链接: [704. 二分查找](704. 二分查找 - 力扣(LeetCode))

【题目描述】

给定一个 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

【思路】

在题目中我们可以注意到在本题中的数组是一个有序数组,同时数组中的所有元素是不重复的,重复的元素往往会导致在返回时获得的下标不唯一。因此在满足以上条件时,我们才可能要考虑使用二分法。

二分法是较为简单的算法,逻辑并不复杂,但是边界值的分析往往会出现问题。我的理解中将这类边界分为两类:左闭右闭、左闭右开。

左闭右闭

即在判断时能达到右边界left<=right,在定义时right=nums.Length-1,同时如果在下文中target<nums[middle]时,right=middle-1

左闭右闭

在判断时无法达到右边界left<right,在定义时right=nums.Length,同时如果在下文中target<nums[middle]时,right=middle


既然已经分析好思路了,上代码!

public 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;
    }
}

这是左闭右闭区间,由于数组有序递增,通过判断nums[middle]与target的大小,来选择向哪一边进行缩小。

但是我们通过提交后可以发现,这个执行的速度并不够快,这时我们可以观察到middle的赋值那里有一个/2操作,经过多年计算机专业知识的熏陶(不是)我们可以想到一个更加高效的方法——位运算!直接上代码

public class Solution {
    public int Search(int[] nums, int target) {
        int left = 0;
        int right = nums.Length;
        while (left < right)
        {
            int middle = left + ( right - left ) >> 1;
            if ( nums[middle] > target)
            {
                right = middle;
            }
            else if ( nums[middle] < target )
            {
                left = middle + 1;
            }
            else
            {
                return middle;
            }
        }
        return -1;
    }
}

哎,这个逻辑上好像没有问题了,但是一提交发现时间超限,这是为什么呢?

通过对运算符优先级的了解,我们发现>>运算符的优先级很低,因此我们只需要加上一个()就可以咯

代码如下

public class Solution {
    public int Search(int[] nums, int target) {
        int left = 0;
        int right = nums.Length;
        while (left < right)
        {
            int middle = left + ( right - left >> 1 );
            if ( nums[middle] > target)
            {
                right = middle;
            }
            else if ( nums[middle] < target )
            {
                left = middle + 1;
            }
            else
            {
                return middle;
            }
        }
        return -1;
    }
}

轻松()拿下这道入门二分法!


27 移除元素

力扣链接:[27.移除元素](27. 移除元素 - 力扣(LeetCode))

【题目描述】

给你一个数组 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 = [3,2,2,3], val = 3
输出:2, nums = [2,2]
解释:函数应该返回新的长度 2, 并且 nums 中的前两个元素均为 2。你不需要考虑数组中超出新长度后面的元素。例如,函数返回的新长度为 2 ,而 nums = [2,2,3,3] 或 nums = [2,2,0,0],也会被视作正确答案。

【思路】

由于数组在内存中的存储地址是连续的,因此我们无法直接将元素从数组中删除,只能选择将后面的元素向前移动并覆盖。在本题中,我想到了两种解决方案,第一个是暴力循环,第二个是双指针。

暴力法

第一个外层for循环用于遍历每个数组元素,当某个元素与需要删除的值相等时,进入内层循环,在内层循环中将当前下标的下一位元素全部前移一位。这个方法简单暴力,但是时间复杂度达到了o(n²),在数据量很大的时候往往会出现超时的情况。

双指针

双指针的思考角度就是快慢指针,程序开始时两个指针同时向后移动,当nums[fast]!=val时,快慢指针同时向后移动,当nums[fast]==val时,慢指针在本次循环中停止移动,从而达到数组元素前移的目的。

【代码】

暴力法

public class Solution {
    public int RemoveElement(int[] nums, int val) {
        int size = nums.Length;
        for(int i = 0;i < size;i++)
        {
            if(nums[i] == val)
            {
                for(int j = i + 1;j < size ;j++)
                {
                    nums[j - 1] = nums[j]; 
                }
                size--;
                i--;
            }
        }
        return size;
    }
}

在以上代码中可以注意到,我们需要获取数组的初始长度,因为在后续代码运行中可能会让他发生动态变化。

其次在每一次if判断中,都要进行size--,i--操作,因为我们在数组元素前移时数组的长度就会减1,同时外层循环的变量i也需要减1。

双指针

public class Solution {
    public int RemoveElement(int[] nums, int val) {
        int slow = 0;
        int length = nums.Length;
        for(int fast = 0;fast < length;fast++)
        {
            if(nums[fast] != val)
            {
                nums[slow] = nums[fast];
                slow++;
            }
        }
        return slow;
    }
}

双指针的思想在上面已经提过了,这里就不再重复了。


总结

重新把以前抛弃的算法捡起来,还有点熟悉的感觉,多多努力!

posted @ 2023-06-09 17:32  asjdqi  阅读(12)  评论(0)    收藏  举报