力扣最热一百题——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,遥遥领先。

总结

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

相关推荐
无敌昊哥战神4 分钟前
大模型(LLM)推理优化技术全景总结
python·算法·大模型
平行侠12 分钟前
A10 恶劣环境传感器信号仿真与统计检验台
算法
洛水水20 分钟前
【力扣100题】34.二叉搜索树中第K小的元素
c++·算法·leetcode
_深海凉_28 分钟前
LeetCode热题100-翻转二叉树
算法·leetcode·职场和发展
吃好睡好便好39 分钟前
在Matlab中绘制抛物三维曲面图
开发语言·人工智能·学习·算法·matlab·信息可视化
伯远医学1 小时前
Nat. Methods | 邻近标记技术:活细胞中捕捉分子互作的新利器
java·开发语言·前端·javascript·人工智能·算法·eclipse
刘永鑫Adam1 小时前
Nature Microbiology | 基于TRACS算法的跨多界宏基因组数据菌株水平溯源推演
算法
小O的算法实验室1 小时前
2026年SEVC,面向无人机辅助边缘计算的自适应群体智能算法,深度解析+性能实测
算法·边缘计算·智能算法·智能算法改进
高锰酸钾_1 小时前
计算机网络-网络层-路由算法与路由协议
计算机网络·算法·智能路由器
智者知已应修善业1 小时前
51单片机4按键控制共阳LED霓虹灯切换1整体闪烁2流水下3流水上4间隔闪烁】2023-10-27
c++·经验分享·笔记·算法·51单片机