首页 > 编程语言 >代码随想录算法训练营,29日 | 704. 二分查找,27. 移除元素,977.有序数组的平方,209.长度最小的子数组,59.螺旋矩阵II

代码随想录算法训练营,29日 | 704. 二分查找,27. 移除元素,977.有序数组的平方,209.长度最小的子数组,59.螺旋矩阵II

时间:2024-08-29 20:16:19浏览次数:18  
标签:right nums int 随想录 mid ++ 数组 移除

数组基础
文档讲解︰代码随想录(programmercarl.com)
1.连续空间、相同类型元素
2.元素只能覆盖
3.二维数组的地址连续吗(C++连续,Java不连续)

704. 二分查找
题目链接:704. 二分查找
文档讲解︰代码随想录(programmercarl.com)
视频讲解︰二分查找
日期:2024-08-29

思路:第一反应是想到二分查找的前提:数组升序,无重复元素;第二点是,确定区间是左闭右开还是左闭右闭,这样才能确定是左右大小比较是<、还是<=。
Java代码如下:

//左闭右闭
class Solution {
    public int search(int[] nums, int target) {
        int left = 0, right = nums.length - 1;
        while(left <= right){
            int mid = left + (right - left)/2;
            if(nums[mid] == target){
                return mid;
            }else if(nums[mid] < target){
                left = mid + 1;
            }else{
                right = mid - 1;
            }
        }
        return -1;
    }
}
//左闭右开
class Solution {
    public int search(int[] nums, int target) {
        int left = 0, right = nums.length;
        while(left < right){
            int mid = left + (right - left)/2;
            if(nums[mid] == target){
                return mid;
            }else if(nums[mid] < target){
                left = mid + 1;
            }else{
                right = mid;
            }
        }
        return -1;
    }
}

代码随想录给出一个优化的版本:

class Solution {
    public int search(int[] nums, int target) {
        // 避免当 target 小于nums[0] nums[nums.length - 1]时多次循环运算
        if (target < nums[0] || target > nums[nums.length - 1]) {
            return -1;
        }
        int left = 0, right = nums.length - 1;
        while (left <= right) {
            int mid = left + ((right - left) >> 1);
            if (nums[mid] == target) {
                return mid;
            }
            else if (nums[mid] < target) {
                left = mid + 1;
            }
            else { // nums[mid] > target
                right = mid - 1;
            }
        }
        // 未找到目标值
        return -1;
    }
}

总结:还要再熟悉下更新mid,寻找区间是左闭右开区间,right更新为mid而不是mid - 1,即:下一个查询区间不会去比较nums[middle]。

27. 移除元素
题目链接:27. 移除元素
文档讲解︰代码随想录(programmercarl.com)
视频讲解︰移除元素
日期:2024-08-29

思路:1.暴力解法,两层for,一层遍历数组找要移除的目标值,另一层是找到移除的值后,将其后面的值移前一位;2.快慢指针:快指针快在它会先走,遍历整个数组,慢指针:如果快指针遇到目标值了,它就不动了,否则将快指针的值给它,这样的好处就是,两个指针都从开头起步,能很方便的找到目标值并用后面的值覆盖。
Java代码如下:

//暴力
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];
                }
                i--;
                size--;
            }
        }
        return size;
    }
}
//快慢指针
class Solution {
    public int removeElement(int[] nums, int val) {
        int slow = 0;
        for(int fast = 0; fast < nums.length; fast++){
            if(nums[fast] != val){
                nums[slow++] = nums[fast];
            }
        }
        return slow;
    }
}

代码随想录还提供了用相向双指针的算法,我用的是同向的:

//相向双指针法
class Solution {
    public int removeElement(int[] nums, int val) {
        int left = 0;
        int right = nums.length - 1;
        while(right >= 0 && nums[right] == val) right--; //将right移到从右数第一个值不为val的位置
        while(left <= right) {
            if(nums[left] == val) { //left位置的元素需要移除
                //将right位置的元素移到left(覆盖),right位置移除
                nums[left] = nums[right];
                right--;
            }
            left++;
            while(right >= 0 && nums[right] == val) right--;
        }
        return left;
    }
}

总结:第一次用双指针的地方,之前面试有被问过什么时候用双指针的地方,后面遇到多了再总结。总的来说就是一边循环里干两遍循环的事。

977.有序数组的平方
题目链接:977.有序数组的平方
文档讲解︰代码随想录(programmercarl.com)
视频讲解︰有序数组的平方
日期:2024-08-29

思路:1.暴力思路,平方后再快排; 2.注意到数组是有序的,那么对于平方后的最大数只会在头或者尾产生,所有两个指针一个在头一个在尾平方比大小,把较大的值放到一个新数组的末尾就行了。
Java代码如下:

class Solution {
    public int[] sortedSquares(int[] nums) {
        int[] result = new int[nums.length];
        int left = 0;
        int right = nums.length - 1;
        int i = nums.length - 1;
        while(left <= right){
            if(nums[left] * nums[left] <= nums[right] * nums[right]){
                result[i--] = nums[right] * nums[right];
                right--;
            }else{
                result[i--] = nums[left] * nums[left];
                left++;
            }
        }
        return result;
    }
}

总结:注意一定要是有序的数组才能这么操作,面试被坑过(

209.长度最小的子数组
题目链接:209.长度最小的子数组
文档讲解︰代码随想录(programmercarl.com)
视频讲解︰长度最小的子数组
日期:2024-08-29

思路:看题第一眼就是滑动窗口(总算记住了),数组跟着前指针的移动逐加,直到和大于等于目标,记录此时子数组长度,此时移动一位后指针,前指针再接着走,接着加,再出现大于等于目标时比较之前的长度和这一次的,取最小值,最后返回时还要判断到底存不存在这个子数组
Java代码如下:

class Solution {
    public int minSubArrayLen(int target, int[] nums) {
        int result = Integer.MAX_VALUE;
        int i = 0, sum = 0;
        for(int j = 0; j < nums.length; j++){
            sum += nums[j];
            while(sum >= target){
                result = Math.min((j - i + 1), result);
                sum -= nums[i++];
            }
        }
        return result == Integer.MAX_VALUE? 0: result;
    }
}

总结:不熟悉Java语法,Integer.MAX_VALUE,Math.min,滑动窗口这名字还是很形象的,主要用在早最小区间。

59.螺旋矩阵II
题目链接:59.螺旋矩阵II
文档讲解︰代码随想录(programmercarl.com)
视频讲解︰螺旋矩阵II
日期:2024-08-29

思路:首先是找不变量,在本题中就是区间的划分,上下左右都得是同样的区间(左闭右开)才能清楚明白的写下去,还有就是完成这套题需要一些什么值要搞清楚,循环起始位置(怎么调整它:需要一个调整值),循环次数,填的数(从1开始),最后就是这个二维矩阵的中心有没有,确定了这些思路就清楚了
Java代码如下:

class Solution {
    public int[][] generateMatrix(int n) {
        int result[][] = new int[n][n];
        int startx = 0, starty = 0;
        int loop = n / 2;
        int mid = n / 2;
        int offset = 1;
        int count = 1;
        while(loop-- > 0){
            int i = startx;
            int j = starty;
            for(; j < n - offset; j++){
                result[i][j] = count++;
            }
            for(; i < n - offset; i++){
                result[i][j] = count++;
            }
            for(; j > 0; j--){
                result[i][j] = count++;
            }
            for(; i > 0; i--){
                result[i][j] = count++;
            }
            startx++;
            starty++;
            offset++;
        }
        if(n % 2 != 0) result[mid][mid] = n * n;
        return result;
    }
}

总结:错在了对二维数组两个角标的理解上,还有这里每个值的意义要搞明白。

标签:right,nums,int,随想录,mid,++,数组,移除
From: https://www.cnblogs.com/wowoioo/p/18387289

相关文章

  • 求数组的最大值-Day2
    求数组[12,1,2,3000,130,31,12,200,500]最大值:声明并保存一个最大元素的变量max默认max取数组第一个元素遍历数组,将每个元素与max做大小比较如果该元素大于max,将这个元素存到max里,否则下一轮最后输出maxletarr=[12,1,2,3000,130,31,12,200,500]l......
  • 代码随想录算法day2-数组2
    题目1209.长度最小的子数组给定一个含有n个正整数的数组和一个正整数target。找出该数组中满足其总和大于等于target的长度最小的子数组[numsl,numsl+1,...,numsr-1,numsr],并返回其长度。如果不存在符合条件的子数组,返回0。示例1:输入:target=7,nums=[2,......
  • 代码随想录day44 || 1143 最长公共子序列, 1035 不相交的线, 53 最大子序列和, 392 判
    1143最长公共子序列funclongestCommonSubsequence(text1string,text2string)int{ //思路和判断最长公共数组一样 //dp[i][j]表示以text1[i],text2[j]为结尾的最长公共子序列的长度 //iftext1[i]==text2[j]{dp[i+1][j+1]=dp[i][j]+1}else{dp[i+1][j+1]=......
  • C语言涉及问题(文件IO与数组和指针)
    一、文件IO相关1、标准出错、输入、输出三者的缓冲机制是什么?标准出错(stderr):属于不缓冲机制,数据直接写入设备标准输入(stdin)和标准输出(stdout):属于行缓冲和全缓冲,行缓冲时需要用'\n'分隔,全缓冲是缓冲区满才会写入或者输出。冲刷缓冲函数:fflush();无论是如果想将全缓冲......
  • 代码随想录算法训练营第四十三天 | 300.最长递增子序列 , 674. 最长连续递增序列 , 718.
    目录300.最长递增子序列 思路1.dp[i]的定义2.状态转移方程3.dp[i]的初始化4.确定遍历顺序 5.举例推导dp数组方法一:动态规划方法二:贪心心得收获 674.最长连续递增序列思路动态规划1.确定dp数组(dptable)以及下标的含义2.确定递推公式3.dp数组如何初始化4.......
  • 代码随想录算法训练营第二十九天(贪心 三)
    力扣题部分:134.加油站题目链接:.-力扣(LeetCode)题面:在一条环路上有 n 个加油站,其中第 i 个加油站有汽油 gas[i] 升。你有一辆油箱容量无限的的汽车,从第 i 个加油站开往第 i+1 个加油站需要消耗汽油 cost[i] 升。你从其中的一个加油站出发,开始时油箱为......
  • Java中的数组用法(复制、替换、查找与排序)
    在Java编程中,数组是一种基础且强大的数据结构,用于存储一组相同类型的元素。本文将深入探讨数组在Java中的用法,并展示如何进行数组的复制与替换、查找以及排序。(这些了解与学习只需要一个IDEA就可以进行练习了 )##数组的声明与初始化在Java中,数组的声明和初始化非常直观。以......
  • leetcode215. 数组中的第K个最大元素,小根堆/快排思想
    leetcode215.数组中的第K个最大元素给定整数数组nums和整数k,请返回数组中第k个最大的元素。请注意,你需要找的是数组排序后的第k个最大的元素,而不是第k个不同的元素。你必须设计并实现时间复杂度为O(n)的算法解决此问题。示例1:输入:[3,2,1,5,6,4],k=2......