关于数据结构中线性表的顺序存储结构删除相同的数据

首先,我们的顺序表是由一个存放数据的数组和一个存放数组最后的一个元素的下标也就是last构成。

具体形式如图

如图,last指向顺序表最后一个元素,我们可以看出此时顺序表中有两个数是有重复的,但我们在实际运算的过程中可能并不知道,我们可以对顺序表进行遍历,定义一个i当作被比较数,定义一个j为比较数,当然j肯定是在i后面一位,因为i没有必要和前面的数以及自己比较,我们是向后面进行遍历的,那么,我们的思路是,i首先指向第一个数据,j也就指向第二个数据,然后将i与j的数据进行比较,如果i与j的数据不相等,也就是说它们不重复,不是我们要删除的对象,我们可以让i不变,j进行++,使其指向后面一位数据,直到j指向最后一个数据,也就是last。

如图,

当j一直++到与last相等也就是j指向最后一个数据时,如果i与j的数据仍然不相等的话,那我们可以知道在整个顺序表中,没有一个数据与i所指向的数据相等,我们可以不考虑它了,于是将i进行++,使其指向下一位,然后再重复上述步骤,继续与后面的数据比较,直到i的数据与j的数据相等时,如图

我们不难看出在进行一轮遍历后,i指向的数据为2,而j指向的数据也为2,它们重复,这时候我们可以开始删除元素了,我们有两个选择,第一是删除i所指,第二是删除j所指,那么在这里有什么区别呢,不妨想象一下,假如我们删除的是i,那么i后面的所有元素全部往前移动一位,如图所示

代码为

if(L->data[i] == L->data[j])         //如果i与j的数据相同
    {
        for(int temp = i;temp < L->last;temp++)     //将i后面的数据全部往前移一位
            {
                L->data[temp] = L->data[temp+1];
            }
            L->last--;           //之后将last--,因为last始终指向最后一位
    }

那么,i的删除就“完成了”,我们不妨接着往下看,看看我们是否大意

将i所指向的数据删除后,实际上现在i指向的是之前j的数据,而j所指向的是之前j+1的数据,这里因为将后面的数据都往前移一位,所以数组的下标产生了变化。随后j再进行++,知道j又指向了与i所指向数据相等的数据,和之前一样,我们继续把i所指向的数据删除,如图

这里依然是数组的下标发生了改变,那么我们在判断j所指向的数据5与i所指向的2不一致后,一轮遍历结束,那么我们的i进行++!!!!!j依然为i+1!!!!!!

关键就在这,在i所指向数据删除后,i指向了被删除数据的下一个数据,也就是上图的3,但是在j遍历完发现没有数据与i所指数据相等后,i进行了++,如图

我们可以看到i跳过了3,于是我们将会输出1,3,3,4,2,5

这显然不是我们所希望的,很多同学(包括我),就在这里出了一个bug

那我们如何避免呢,我们可以这样

if(L->data[i] == L->data[j])         //如果i与j的数据相同
    {
        for(int temp = i;temp < L->last;temp++)     //将i后面的数据全部往前移一位
            {
                L->data[temp] = L->data[temp+1];
            }
            L->last--;           //之后将last--,因为last始终指向最后一位
            i--;                 //将i“归位”
    }

与上一段代码相比呢,多了一句i--,那么它的作用就在于“复位”,也就是阻止其指向下一位,因为这一位被删除了,我们要让下一次遍历i所指向的仍然是被删除的下一位,而不是跳过它,如图:

当然,我们也可以直接删除j

if(L->data[i] == L->data[j])         //如果i与j的数据相同
    {
        for(int temp = j;temp < L->last;temp++)     //将j后面的数据全部往前移一位
            {
                L->data[temp] = L->data[temp+1];
            }
            L->last--;           //之后将last--,因为last始终指向最后一位
    }

好的,我们今天的分享就到这里,感谢大家于百忙之中得到微之甚微的知识,下面附赠顺序表的源码,只有简单的增删查改接口,其余的大家一起动手来实现吧!

#include<stdio.h>
#include<stdlib.h>
#include"shun.h"
#include<string.h>
SqLink InitList()  //顺序表初始化
{
     SqLink L;
    L = (SqLink)malloc(sizeof(SqList)); //申请地址
    if(L == NULL){
        printf("malloc error!\n");
        return L;
    }
    memset(L,0,sizeof(SqList));   //将长度为sizeof(SqList)的顺序表元素全部赋值为0
    L->last = -1;
    return L;
}
int List_Show(SqLink L)  //顺序表的展示
{
    if(L == NULL||L->last == -1){
        printf("Nothing can show!\n");
        return -1;
    }
    for(int i = 0;i<=L->last;i++)
    {
        printf("%d ",L->data[i]);
    }
    puts("");
    return 0;
}
int List_Insert(SqLink L,int sub,DataType num) //顺序表的插入
{
    if(L == NULL){
        printf("List NULL!\n");
        return -1;
    }
    if(L->last == MAXLEN-1){
        printf("List Full!\n");
        return -1;
    }
    if(sub<0||sub>L->last+1){
        printf("sub error!\n");
        return -1;
    }
    if(L->last == -1)
    {
        L->last++;
        L->data[L->last] = num;
        return 0;
    }
    for(int i = L->last;i>=sub;i--){
        L->data[i+1] = L->data[i];
    }
    L->data[sub]  = num;
    L->last++;
    return 0;
}
int List_Delete(SqLink L,int sub) //顺序表的删除
{
    if(L == NULL||L->last == -1){
        printf("List Nothing can delete!\n");
        return -1;
    }
    for(int i = sub;i<L->last;i++)
    {
        L->data[i]  =L->data[i+1];
    }
    L->last--;
    return 0;
}
int List_Find(SqLink L,DataType num) //查找顺序表中的某一元素,可能为多个
{    
    int temp = 0;
    if(L == NULL||L->last == -1){
        printf("List is NULL!\n");
        return -1;
    }
    for(int i = 0;i<=L->last;i++)
    {
        if(L->data[i] == num){
            printf("num is %d,sub is %d\n",L->data[i],i);
            temp++;
        }
    }
    if(temp == 0)
        printf("Not fount!\n");
    return 0;
}
int List_Change(SqLink L,int sub,DataType num) //顺序表的修改
{    
    if(L == NULL||L->last == -1){
        printf("Nothing can change!\n");
        return -1;
    }
    L->data[sub] = num;
    return 0;
}
int List_Norepist_Norep(sq L)  //删除相同元素
{
     if(L == NULL){
         printf("error\n");
         return 0;
     }
    for(int i = 0;i<L->last;i++)
    {
        for(int j = i+1;j<=L->last;j++)
        {
            if(L->data[i] == L->data[j])
            {
                for(int temp = i;temp<L->last;temp++)
                {
                    L->data[temp] = L->data[temp+1];
                }
                L->last--;
                i--;
            }
        }
    }
     return 0;
}
    
  • 2
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值