这个题目是一个相对简单的题,解决的方法有很多,比如两层for循环的暴力解法,双指针法等。
先来看一下题目和例子
有的同学可能说了,多余的元素,删掉不就得了。
要知道数组的元素在内存地址中是连续的,不能单独删除数组中的某个元素,只能覆盖。
而且这题的要求是空间复杂度为O(1),也就是原地移除。
原地移除可能有点难。我们先假设可以另外创建一个数组,也是实现原地移除的一个前提基础思想。
思路1:
我们开辟一个相等大小的数组arr2,然后两个指针分别指向原数组arr1和新数组arr2。然后arr1开始向后遍历,如果arr【i】!=val就将其值赋给arr2【j】,然后j++往后,直到arr1被遍历完
思路有了,那么我们写代码也就水到渠成了。按照思路一步一步写出来即可
int main()
{
int arr1[4] = { 1,1,2,1 };
int arr2[4] = { 0 };
int i = 0,j = 0,sz=sizeof(arr1)/sizeof(arr1[0]),val=2;
for (i = 0;i < sz;i++)
{
if (arr1[i] != val)
{
arr2[j++] = arr1[i];
}
}
for (i = 0;i < sz;i++)
{
printf("%d ", arr2[i]);
}
printf("\n");
return 0;
}
运行结果截图:
思路2:
虽然思路1可以完成,但是达不到题目的原地移除的要求,不过我们可以模仿思路1进行改造——如果那个新的数组就是原数组本身呢?
我们先定义一个fast指针,负责遍历所有的原数组;定义一个slow指针,负责指向新数组(此时新数组就是数组本身)。遇到不等于val的元素就将其赋值给新数组arr【slow】,然后slow++,直到遍历完原本的数组,新数组也完成了,我们也就实现了原地移除。
将arr[fast]赋给此时的arr[slow],也就完成了移除
有了思路,写起来也就相当顺利
int removeElement(int* nums, int numsSize, int val)
{
int fast=0,slow=0,sz=0;
for(fast=0;fast<numsSize;fast++)//快指针遍历一次数组,慢指针指向的是新数组的位置
{
if(nums[fast]!=val)//不为要删除的元素,将他并入新数组中
{
nums[slow++]=nums[fast];
sz++;
}
}
return sz;
}
以上的方法就是双指针法,要体会到双指针法的妙用。其在刷题的过程中双指针还是十分常见的,我们应当将其牢牢把握。
以后博主也会分享其他运用双指针法的题,希望大家多多支持哦!