力扣最热一百题——4.移动零

目录

日常吐槽

[题目链接:283. 移动零 - 力扣(LeetCode)](#题目链接:283. 移动零 - 力扣(LeetCode))

题目描述

示例

提示

解法一:交换位置

思路

代码实现

解法二:覆写元素

思路

代码实现

总结



日常吐槽

今天本来是计划玩玩游戏的,结果我直接上头了,输一把赢一把的,唉,最后也没法从完美B+差三分上A,还到欠了四十多分才上A,我真的服了,话不多说直接上今天的题目吧。


题目链接:283. 移动零 - 力扣(LeetCode)

注:下述题目描述和示例均来自力扣

题目描述

给定一个数组 nums,编写一个函数将所有 0 移动到数组的末尾,同时保持非零元素的相对顺序。

请注意 ,必须在不复制数组的情况下原地对数组进行操作。

示例

示例 1:

复制代码
输入: nums = [0,1,0,3,12]
输出: [1,3,12,0,0]

示例 2:

复制代码
输入: nums = [0]
输出: [0]

提示

  • 1 <= nums.length <= 104
  • -231 <= nums[i] <= 231 - 1

**进阶:**你能尽量减少完成的操作次数吗?

呃不能。。。。。


这里题目有一个很关键的信息就是要在不复制数组的情况下对数组进行操作,也就是我们要在原数组上进行操作,不能新开一个数组,否则就太简单了,直接遍历一遍不要0就行了。并且记住要保持其他元素的相对顺序。

解法一:交换位置

思路

这个解法的思路就是我每次先找到一个0,这个0的位置是最靠前的一个0,然后在这个0的位置上往后找第一个不为零的元素,使用两个指针记录这两个数的位置,交换他们的位置即可,非常简单,思路也是很清晰的,交换完最后一个0之后,所有的0也就在后面了。

代码实现

java 复制代码
class Solution {
    public void moveZeroes(int[] nums) {
        // 0,1,0,3,12
        // l r
        // 1,0,0,3,12
        //  lr 
        // 1,3,12,0,0
        //      l   r
        // 获取数组的长度
        int len = nums.length;
        // 定义出两个指针
        int left = 0;
        int right = 1;
        // 进入寻找非0元素的过程
        // left记录0的位置,right记录第一个遇到的非零的位置
        while(left < len){
            // 结束特判
            if(right == len && nums[right - 1] == 0){
                break;
            }
            if(nums[left] == 0){
                while(right < len){
                    if(nums[right] != 0){
                        swap(nums,left,right);
                        left++;
                        right = left + 1;
                        break;
                    }else{
                        right++;
                    }
                }
            }else{
                left++;
                right++;
            }
        }
    }


    /**
        交换数组位置
    */
    public void swap(int[] arr,int left,int right){
        int temp = arr[left];
        arr[left] = arr[right];
        arr[right] = temp;
    }
}

说实话,虽然说这个方法比较好思考出来,但是利于人脑思考的方法最终的效果都是不尽人意的。


解法二:覆写元素

思路

这个解法的思路是将数组中的所有0移动到数组的末尾,而保持非零元素的相对顺序不变。

首先,定义两个指针 left 和 right,初始时都指向数组的开头。right 指针用于遍历数组,left 指针用于记录下一个非零元素应该存放的位置。

然后,开始遍历数组,当 right 指针指向的元素为0时,right 指针向右移动一位,跳过该次循环。这是因为我们要将所有的0移动到数组的末尾,所以需要忽略所有的0。

当 right 指针指向的元素不为0时,将该元素覆盖到 left 指针指向的位置,并将 left 指针向右移动一位,即 left++。这样,非零元素就被移动到了正确的位置上。

循环结束后,left 指针后面的元素都应该是0,所以我们再次遍历数组,将 left 指针后面的元素都设置为0。

最终的结果就是将所有的0移动到了数组的末尾,而非零元素的相对顺序保持不变。

代码实现

java 复制代码
class Solution {
    public void moveZeroes(int[] nums) {
        int left = 0;
        int right = 0;
        while(right < nums.length){
            if(nums[right] == 0){
                //这里就是right指针遇到0了
                //需要跳过这个数和跳过这次循环
                right++;
                continue;
            }
            //这里进行覆盖操作
            nums[left++] = nums[right++];
        }
        while(left < nums.length){
            nums[left++] = 0;
        }
    }
}

这个方法的执行效率和内存开销直接起飞了bro,遥遥领先。

总结

官方题解我觉得写得非常的难以理解,但是他确实是把交换的思路优化到极致了,以至于执行效率还是很快,所以这个世界上的万千代码和人一样,厉害的,你更难以洞察其内心。

相关推荐
YGGP6 小时前
【Golang】LeetCode 42. 接雨水
算法·leetcode·职场和发展
仰泳的熊猫6 小时前
题目1466:蓝桥杯基础练习VIP-字符串对比
数据结构·c++·算法·蓝桥杯
Renhao-Wan6 小时前
Java算法实践(二):堆与PriorityQueue实战
java·数据结构·算法
每天要多喝水6 小时前
动态规划Day29:打家劫舍
算法·动态规划
IT猿手6 小时前
多目标鲸鱼优化算法(MOWOA)求解46个多目标函数及一个工程应用,包含四种评价指标,MATLAB代码
开发语言·算法·matlab
罗湖老棍子6 小时前
星际能量矩阵:树形 DP 的递归与非递归双解
算法·动态规划·dfs·bfs·树型dp·树型动态规划
@––––––6 小时前
力扣hot100—系列7-二分查找
数据结构·算法·leetcode
MicroTech20256 小时前
微算法科技(NASDAQ: MLGO)引入量子启发式算法与区块链融合的数据预测与安全传输方案
科技·算法·启发式算法
近津薪荼7 小时前
优选算法——前缀和(5):和为 K 的子数组
算法
你撅嘴真丑7 小时前
第九章-竞赛题目选讲-跳舞机
数据结构·算法