LeetCode - 1089. 复写零

题目

1089. 复写零 - 力扣(LeetCode)

思路

这道题我首先想到的是从前往后双指针,但是这样做会造成数据的覆盖,比如说下面的这个情况

所以解决的方法就是从后往前去复写,但是从后往前的话就要知道最后一个有效元素是什么。

对于这个情况,我们可以先去根据"异地操作"找到最后一个有效元素,需要一个cur指针,和一个dest指针,cur先进行判断,当遇到非0元素cur++,dest++,当遇到0元素,就让cur,dest+2,直到dest到数组的末端,判断此时的cur就是最后一个有效元素

再进行优化,我们就需要双指针下的"就地"操作

但是此时会有个特殊情况:当cur最后一个位置是0的时候,会导致dest+2的时候越界了,此时就会报错了

所以我们要处理这个边界情况,方法就是当查找有效位置的时候,跳出循环后,如果dest=n,此时说明cur指向的一定0(这样另外数的dest连跳两步所以才会越界),所以我们此时让dest-1的位置为0,然后dest-=2,cur-1

接下来就是

读者可能出现的错误写法

cpp 复制代码
class Solution {
public:
    void duplicateZeros(vector<int>& arr) {
        int n = arr.size();
        int cur = 0;
        int dest = 0;

        while(arr[dest])
        {
            if(arr[cur]==0)
            {
                cur++;
                dest+=2;
            }
            else
            {
                cur++;
                dest++;
            }
        }

        while(arr[cur])
        {
            if(arr[cur]==0)
            {
                cur--;
                for(int i=0;i<2;i++)
                {
                    arr[dest] = 0;
                    dest--;
                }

            }
            else
            {
                arr[dest] = arr[cur];
                cur--;
                dest--;
            }  
        }

    }
};

第一个问题是循环条件写错了,while(arr[dest]) 这里有问题,你这样写的话,如果dest越界了,访问arr[dest]就会出错。应该改成 while(dest < n)。

第二个问题是第二个循环也有问题,while(arr[cur]) 这里也不对,应该改成 while(cur >= 0)。

第三个问题是缺少边界处理,你没有处理那种刚好在边界的特殊情况。

修改一下:首先第一个while循环应该是 while(dest < n),然后在循环里面如果arr[cur]等于0就让dest加2,否则dest加1,但是要记得加个判断if(dest >= n) break,然后cur++。

接下来要处理边界情况,如果dest==n的话,说明最后一个0只能写一半,所以arr[n-1] = 0,然后dest减2,cur减1。

最后第二个while循环应该是while(cur >= 0),在循环里面如果arr[cur]是0就连续写两个0到dest位置,否则就把arr[cur]写到dest位置,记得每次都要cur和dest往前移。

dest表示"当前元素应该在的最终位置",因为我们不知道第一个元素应该在的位置,所以应该初始化为-1

cpp 复制代码
class Solution {
public:
    void duplicateZeros(vector<int>& arr) {
        int n = arr.size();
        int cur = 0;
        int dest = -1;

        while(dest<n)
        {
            if(arr[cur]==0)
            {
                dest+=2;
            }
            else
            {
                dest++;
            }
            if(dest >= n-1) break;
            cur++;
        }

        if(dest>n-1)
        {
            arr[n-1] = 0;
            dest-=2;
            cur--;
        }

        while(cur>=0)
        {
            if(arr[cur]==0)
            {
                for(int i=0;i<2;i++)
                {
                    arr[dest] = 0;
                    dest--;
                }
            }
            else
            {
                arr[dest] = arr[cur];
                dest--;
            } 
            cur--; 
        }
    }
};
相关推荐
Meme Buoy3 小时前
18.补充数学1:生成树-最短路径-最大流量-线性规划
数据结构·算法
汀、人工智能3 小时前
[特殊字符] 第89课:岛屿数量
数据结构·算法·数据库架构·图论·bfs·岛屿数量
九英里路3 小时前
cpp容器——string模拟实现
java·前端·数据结构·c++·算法·容器·字符串
2401_892070984 小时前
顺序栈(动态数组实现) 超详细解析(C++ 语言 + 可直接运行)
数据结构·c++·顺序栈
漫霂4 小时前
二叉树的翻转
java·数据结构·算法
3秒一个大4 小时前
深入理解 JS 中的栈与堆:从内存模型到数据结构,再谈内存泄漏
前端·javascript·数据结构
旖-旎5 小时前
哈希表(存在重复元素)(3)
数据结构·c++·学习·算法·leetcode·散列表
计算机安禾5 小时前
【数据结构与算法】第39篇:图论(三):最小生成树——Prim算法与Kruskal算法
开发语言·数据结构·c++·算法·排序算法·图论·visual studio code
汀、人工智能5 小时前
[特殊字符] 第77课:最长递增子序列
数据结构·算法·数据库架构·图论·bfs·最长递增子序列
澈2075 小时前
堆排序:高效构建大顶堆实战
数据结构·算法·排序算法