LeetCode.26,27,88三题-双指针的运用

本文将对3道解决方法类似的题目进行逐一分析,这三道题目分别是:

LeetCode.26 删除有序数组中的重复项

LeetCode.27 移除元素

LeetCode.88 合并两个有序数组

1. LeetCode.27 移除元素:

题目内容如下:

 假设一个数组nums为:

nums = [0,1,2,2,3,0,4,2]

元素val = 2.

按照题目的要求,需要移除数组nums中所有等于2的元素。对于此题的解析,文章提供三种参考思路

1.在之前处理顺序表中删除元素的问题时,采用的方法是将目标元素后面的元素全部向前覆盖一位。但是,这种处理方法的时间复杂度为O(N^2)过于缓慢。

2.创建一个指针和一个新的数组,遍历整个数组,在便利的过程中。若被遍历的元素大小不等于val时,将此元素复制到新开辟的数组中。当被遍历的元素大小等于val时,不复制并且将指针指向下一个元素。此方法的时间复杂度为O(N),相对于方法1,大幅度降低了程序执行所用的时间。但是,该方法因为新创建了一个数组。空间复杂度为O(N),不符合题目中的要求。

3. 虽然方法2不满足题目的要求,但是可以通过方法2的思路来延伸出方法3,也就是文章题目中提到的双指针法

对于本题,双指针法的具体用法如下:

1.创建两个指针,这里创建的指针分别命名为src,dst。最开始,两个指针都指向数组首元素的下标,即:

 对于本题中创建的两个指针的动作,总结下来就是:将数组中src位置上不等于val的元素放置到dst中。

当两个指针所对应的元素相等且不等于val时,都指向下一个元素。

当指针src对应的值等于val时,指针src指向下一个位置,dst不移动

当两个指针对应的元素不同且不等于val时,将指针src对应的元素赋值到dst位置。并且都向前移动一位。

用图片表示下列过程,即:

1.

 2.

此时, 指针src对应的值等于val。所以指针dst不动,指针src继续向后移动:

此时,指针 src对应的值不等于dst,并且不等于val。所以,进行赋值操作:

 再次向后移动,还会重复上面的动作:

当 src遍历整个数组后,数组中内容如下图所示:

上述过程对应的代码为:

int removeElement(int* nums, int numsSize, int val){
    int dst = 0;
    int src = 0;
    while( src < numsSize)
    {
        if( nums[src] != val)
        {
            nums[dst++] = nums[src++];
        }
        else
        {
            src++;
        }
    }
    return dst;

}

 2. LeetCode.88 合并两个有序数组:

题目如下:


 

给出的示例如下:
 

 用图片表示上面给出的示例,即:

 对于这道题的解法,依旧采用双指针的思想,对于每一个数组均创建一个指针。但是,如果再次采用上面从头到尾进行遍历的方法,如果再某处需要插入元素,则还是会出现顺序表中插入元素出现的问题,即:每插入一个元素,都需要将后面的元素整体后移动一位。所以对于此题。最好采用从后向前,从大到小的顺序进行遍历。并且,将第一个数组中最大的元素与第二个数组中的元素分别进行比较。较大的则插入到第一个数组中后面的区域,将上述过程用图片演示,即:

 上面的情况中,是第二个数组元素全部遍历完成时,第一个数组中的元素没有完成遍历。但是对于下面的情况,即:第一个数组完成遍历时,第二个数组并没有完成遍历:

用图片表示上述数组中元素的变化情况:

 ​​

 

 最后的结果如上图所示: 第二个数组中的元素2并没有插入到第一个数组中。并且第一个数组已经遍历完成。而第二个数组没有遍历完成。

总结上述过程,为了方便陈述,将第一个数组命名为nums1,将第二个数组命名为nums2

nums1创建的指针命名为dst,为nums2创建的指针命名为src

从后向前对两个数组同时遍历,

当满足src对应的元素>dst对应的元素时,将src对应的元素在nums1中进行一个尾插操作。并且两个指针均指向前一个元素。

当不满足上述关系时,将dst对应的元素在nums1在前面插入元素的基础上进行一个尾插操作。并将dst指向前一个元素。

nums2遍历完成时。标志着程序运行完成。当nums1遍历完成但是nums2没有完成遍历时。将nums2剩余的元素在nums1中进行插入。

用代码表示上述过程:

首先,题目中已给的参数分别是:

m表示 nums1中非0的元素。n表示nums2中的元素。为了方便表示,用end1,end2表示上面的参数。end表示nums1中加上0元素的总长度。

代码如下:

void merge(int* nums1, int nums1Size, int m, int* nums2, int nums2Size, int n){
    int end1 = m - 1; 
    int end2 = n -1;
    int end = m + n -1;
   while( end1 >= 0 && end2 >= 0)
   {
       if( nums1[end1] > nums2[end2])
       {
           nums1[end--] = nums1[end1--];
       }
       else
       {
           nums1[end--] = nums2[end2--];
       }
   }
       while( end2 >= 0)
       {
           nums1[end--] = nums2[end2--];
       }

}

3. LeetCode. 26 删除有序数组中的重复项:

题目如下:

 示例如下:

依旧采用双指针的方法来结局问题。创建两个指针分别为src,dst

从上面的题目要求可知。题目涉及到对于数组中元素的更改。所以,创建的两个指针在开头可以处于相同位置,也可以处于不同位置。为了方便演示,这里给出不同位置的情况,例如下面图片所示的数组:

 

 因为要保证数组中元素不能重复,所以,需要将后面的元素覆盖到前面元素的位置。所以,需要一个指针取读取后一位的元素,并且与前一位的元素进行对比。此时如果后一位元素与本位元素相同。则src指向后一个元素:

 此时,两个指针指向的元素不同,所以,先让dst指向后一位元素,再将src中的元素赋值给此时的dst,再src+1,此时两个指针指向的元素相同,所以一直src+1,直到:

 重复上面所说的步骤,即:

 

 按照前面说的规则,最后的效果为

 

通过图片不难发现,程序结束的标志就是当指针src <= 数组中元素个数-1时(或者src <数组中元素个数时)。

总结上述过程,可以分为三个要点:

1. src < 数组中元素个数时,程序结束

2.src指向的值=dst时,src指向后面的元素。dst不变。

3. src指向的值不等于dst时,先将dst+1,再将src+1指向的值赋值到dst,最后src+1.

用代码表示,即:

int removeDuplicates(int* nums, int numsSize){
    int dst = 0;
    int src = 1;
  while( src < numsSize)
  {
      if( nums[src] == nums[dst])
      {
          src++;
      }
      else
      {
          dst++;
          nums[dst] = nums[src];
          src++;
      }
  }
    return dst+1;

}

 


 

 

  • 3
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

起床写代码啦!

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值