前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >[优选算法]——双指针——Leetcode——1089. 复写零

[优选算法]——双指针——Leetcode——1089. 复写零

作者头像
小李很执着
发布2024-06-15 09:37:24
420
发布2024-06-15 09:37:24
举报
文章被收录于专栏:学习笔记学习笔记

1.题目

1089. 复写零

给你一个长度固定的整数数组 arr ,请你将该数组中出现的每个零都复写一遍,并将其余的元素向右平移。

注意:请不要在超过该数组长度的位置写入元素。请对输入的数组 就地 进行上述修改,不要从函数返回任何东西。

示例 1:

代码语言:javascript
复制
输入:arr = [1,0,2,3,0,4,5,0]
输出:[1,0,0,2,3,0,0,4]
解释:调用函数后,输入的数组将被修改为:[1,0,0,2,3,0,0,4]

示例 2:

代码语言:javascript
复制
输入:arr = [1,2,3]
输出:[1,2,3]
解释:调用函数后,输入的数组将被修改为:[1,2,3]

提示:

  • 1 <= arr.length <= 104
  • 0 <= arr[i] <= 9

2.解法(原地复写-双指针): 1.算法思路:

如果「从前向后」进⾏原地复写操作的话,由于? 0 ?的出现会复写两次,导致没有复写的数「被覆盖掉」。因此我们选择「从后往前」的复写策略。但是「从后向前」复写的时候,我们需要找到「最后⼀个复写的数」,因此我们的⼤体流程分两步: i. 先找到最后⼀个复写的数;(最重要的一步) ii. 然后从后向前进⾏复写操作。

2.算法流程:

a. 初始化两个指针 cur = 0 , dest = 0 ;

b. 找到最后⼀个复写的数: i. 当cur < n 的时候,⼀直执⾏下⾯循环: • 判断 cur 位置的元素: ◦ 如果是 0 的话, dest 往后移动两位; ◦ 否则, dest 往后移动⼀位。 • 判断?dest ?时候已经到结束位置,如果结束就终⽌循环; • 如果没有结束, cur++ ,继续判断。 c. 判断dest 是否越界到 n 的位置: i. 如果越界,执⾏下⾯三步: 1. n - 1 位置的值修改成0 ; 2. cur 向移动⼀步; 3. dest 向前移动两步。

d. 从cur 位置开始往前遍历原数组,依次还原出复写后的结果数组: i. 判断cur 位置的值: 1. 如果是 0 : dest 以及 dest - 1 位置修改成 0 , dest -= 2 ; 2. 如果⾮零: dest 位置修改成0 , dest -= 1 ; ii. cur-- ,复写下⼀个位置。

3.手撕流程图

先根据“异地”操作,然后优化成双指针下的“就地”操作

找到最后⼀个复写的数:

4.代码实现

1.C语言

代码语言:javascript
复制
void duplicateZeros(int* arr, int arrSize)
{
    //先找到最后一个复写数
    int cur=0,dest=-1;
    while(cur<arrSize)
    {
        if(arr[cur])
        {
            dest++;
        }
        else
        {
            dest+=2;
        }
        if(dest>=arrSize-1)
        {
            break;
        }
        cur++;
    }
    //处理边界情况
    if(dest==arrSize)
    {
        arr[ arrSize-1]=0;
        cur--;
        dest-=2;
    }
    //3.从后往前完成复写操作
    while(cur>=0)
    {
        if(arr[cur])//不为零是复写一次
        {
            arr[dest--]=arr[cur--];
        }
        else//为零时
        {
            arr[dest--]=0;
            arr[dest--]=0;
            cur--;
        }
    }
}

2.C++

代码语言:javascript
复制
class Solution {
public:
    void duplicateZeros(vector<int>& arr) 
    {
       int cur=0,dest=-1,n=arr.size();
       while(cur<n)
       {
        if(arr[cur])  dest++;
        else  dest+=2;
        if(dest>=n-1)  break;
        cur++;
       }
       if(dest==n)
       {
        arr[n-1]=0;
        cur--;
        dest-=2;
       }
       while(cur>=0)
       {
        if(arr[cur]) arr[dest--]=arr[cur--];
        else
        {
            arr[dest--]=0;
            arr[dest--]=0;
            cur--;
        }
       }
    }
};
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2024-05-06,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 1.题目
  • 2.解法(原地复写-双指针): 1.算法思路:
    • 2.算法流程:
    • 3.手撕流程图
      • 找到最后⼀个复写的数:
      • 4.代码实现
        • 1.C语言
          • 2.C++
          领券
          问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档