顺序表(线性表的顺序存储结构)

逻辑结构上呈线性分布的数据元素在实际的物理存储结构中也同样相互之间紧挨着,这种存储结构称为线性表的顺序存储结构。

也就是说,逻辑上具有线性关系的数据按照前后的次序全部存储在一整块连续的内存空间中,之间不存在空隙,这样的存储结构称为顺序存储结构。

使用顺序存储结构存储的数据,第一个元素所在的地址就是这块存储空间的首地址。通过首地址,可以轻松访问到存储的所有的元素,只要首地址不丢,数据永远能找到(一根绳上的蚂蚱,要有就都有)。

 

顺序表中存放数据的特点和数组这种数据类型完全吻合,所以顺序表的实现使用的是数组

数组实现顺序表的存储结构时,一定要注意预先申请足够大的内存空间,避免因存储空间不足,造成数据溢出,导致不必要的程序错误甚至崩溃。

在建立顺序表时,除了预先申请内存空间,还需要实时记录顺序表的长度和顺序表本身申请的内存大小,便于后期对顺序表中的数据元素进行调去。

所以,要自定义顺序表的结构:

typedef struct Table{
int *head;//声明了一个名为head的长度不确定的数组,也叫动态数组
int length;//记录当前顺序表的长度
int size;//记录顺序表分配的存储容量
}table;

建立顺序表,就是顺序表进行初始化,在预先申请内存空间的同时,给变量size和length赋初值:

table initTable()
{
table t;
t.head = (int *)malloc(Size*sizeof(int));//构造一个空的顺序表,动态申请存储空间
if(!t.head)//如果申请失败,作出提示并直接退出程序
{
    printf("初始化失败");
    exit(0);
}
t.length=0;//空表的长度初始化为0
t.size=Size;//空表的初始化存储空间为Size
return t;

顺序表查找元素-采用顺序查找算法(普通的遍历算法)

//查找函数,其中elem表示要查找的数据元素的值
int selectTable(table t,int elem)
{
    for(int i=0;i<t.length;i++)
    {
        if(t.head[i] == elem)
        {
            return i+1;
        }
    }
    return -1;//如果查找失败,返回-1
}

更改顺序表中数据元素,最简单直接的办法就是:调用查找算法找到该数据元素的位置,直接在该位置上更改。

//更改函数,其中,elem为要更改的元素,newElem为新的数据元素
table amendTable(table t, int elem, int newElem)
{
int add = selectTable(t, elem);
t.head[add-1] = newElem;//由于返回的是数据元素在顺序表中的位置,所以-1就是该元素在数组中的下标,数组的下标从0开始
return t;

顺序表插入元素

1,在表头插入

2,在表的中间某个位置插入

3,直接尾随顺序表,作为表的最后一个元素

无论在顺序表的什么位置插入数据元素,解决办法都是:找到要插入的位置,将后继数据元素 整体向后移动一个位置,最后直接在腾出来的位置上插入数据元素。

//插入函数,其中elem为插入的元素,add为插入到顺序表的位置
table addTable(table t, int elem, int add)
{
//判断插入本身是否存在问题(如果插入元素位置比整张表的长度+1还大(如果相等,是尾随的情况)
//或者插入的位置本身不存在,程序作为提示并自动退出)
if(add>t.length+1 || add<1)
{
printf("插入位置有问题");
return t;
}
//做插入操作时,首先需要看顺序表是否有多余的存储空间提供给插入的元素,如果没有,需要申请
if(t.length==t.size){
t.head=(int *)realloc(t.head,(t.size+1)*sizeof(int));
if(!t.head){
printf("存储分配失败");
return t;
}
t.seze+=1;
}
//插入操作,需要将从插入位置开始的后续元素,逐个后移
for(int i=t.length-1;i>=add-1;i--)
{
t.head[i+1] = t.head[i];
}
//后移完成后,直接将所需插入元素,添加到顺序表的相应位置
t.head[add-1] = elem;
//由于添加了元素,所以长度+1
t.length++;
return t;

在此程序中,当数组存储空间不足时,使用realloc函数每次额外多申请一个int型的存储空间,这么做还不是最优。最好的办法就是每次发现空间不够时,多申请几个内存空间,这么做的好处是:在后续做插入操作过程中不需要每次都运行realloc函数,提高了程序的运行效率

 

顺序表删除元素

在数组中删除元素时,只需将该元素所在位置后的所有元素整体前移1一个位置即可

前移的过程中被删除元素被后一个元素覆盖掉,简介实现了删除操作。

table delTable(table t, int add)
{
if(add>t.length || add<1)
{
printf("被删除元素的位置有误");
exit(0);
}
//删除操作
for(int i=add;i<t.length;i++)
{
t.head[i-1]=t.head[i];
}
t.length--;
return t;

顺序表实现的基础,完全借用了数组这一数据类型,优点是在对数据进行遍历时,数据在连续的物理空间中存放,查找的速度比较快。

但是由于数组本身限制,在向顺序表中新增或者删除元素时,如果被操作位置后需有很多数据元素,后需所有数据元素都需要前移或者后移,效率比较低。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值