欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 新闻 > 会展 > 【数据结构】一文讲解线性表之顺序表概念及其基本操作(附C语言源码)

【数据结构】一文讲解线性表之顺序表概念及其基本操作(附C语言源码)

2024/12/28 4:52:22 来源:https://blog.csdn.net/weixin_58498967/article/details/143636007  浏览:    关键词:【数据结构】一文讲解线性表之顺序表概念及其基本操作(附C语言源码)
前文我们讲过数据结构的三个部分:数据、数据元素和数据结构以及数据结构的三要素:逻辑结构、物理结构和数据运算。现在我们从三个组成部分和三要素讲解

线性表的定义和基本操作

定义

线性表是一个抽象的概念,一般具有相同数据类型的n(n>=0)个数据元素的有限序列,其中n为表长。(是不是感觉类似数组,但数组不是线性表,但也具有线性表的一些特性)

线性表指的是数据元素通过线性逻辑结构连接的一种数据结构。

数据元素之间是线性关系,形如A->B->C…->N

基本概念

  • 空表 n=0,线性表长度为0
  • 位序 线性表中第i个数据元素
  • 表头元素 线性表中第一个数据元素
  • 表尾元素 线性表中最后一个数据元素
  • 数据元素之间存在一对一的线性关系,即每个元素(除第一个外)有且仅有一个前驱,每个元素(除最后一个外)有且仅有一个后继。
基本操作

线性表的基本操作指的是几个基本的数据运算,一般都有如下几个操作:

  • 建立表
  • 初始化表
  • 插入
  • 删除
  • 查找

下面就讲解一下线性表中的线性表中的顺序表的概念以及基本操作

顺序表

线性表定义了数据结构的逻辑结构为线性,而顺序表则定义其物理结构为连续的,顺序的线性表。即用顺序储存的方式实现线性表(逻辑上相邻的元素在顺序上也相邻)

顺序表的建立与初始化
*******************************静态分配******************************************
// 定义线性表的最大长度
#define MAX_SIZE 100// 定义线性表的结构体
typedef struct {ElemType data[MAX_SIZE]; // 存储数据的数组   数组的存储方式为顺序的int length; // 当前线性表的长度 
} SeqList; //定义顺序表的名字// 初始化线性表 
void InitList(SeqList *L) { for(int i=0 ; i < MAX_SIZE ; i++) {L.data[i] = 0 ;//初始化所有元素为0}L.length = 0; // 初始长度为0 
}**特别注意的是ElemType变量,它不是一个实际的数据类型,指代任意一种数据类型,例如int float double等**静态分配的顺序表长度固定,不可拓展,实际应用中不方便,我们更多采用动态分配创建顺序表
*******************************动态分配******************************************
// 定义线性表的默认最大长度
#define MAX_SIZE 100// 定义线性表的结构体
typedef struct {ElemType *data;  // 指示动态分配的指针 ElemType表示任意数据类型(例如int) int length;      // 当前顺序表的长度 int MaxSize;     //顺序表最大容量
} SeqList; // 初始化线性表 
void InitList(SeqList *L) { L.data=(ElemType *)malloc(MAX_SIZE*sizeof(ElemType));//分配连续空间L.length = 0; // 初始长度为0 L.MaxSize = MAX_SIZE;     //顺序表最大容量
}//增加动态数组的长度
void IncreaseSize(SeqList &L,int len){ElemType *p=L.data;L.data=(ElemType *)malloc((L.MAX_SIZE+len)*sizeof(ElemType));for(int i=0 ; i < L.length ; i++){L.data[i]=p[i];}L.MaxSize=L.MaxSize + len;free(p);
}

特别注意的是ElemType变量,它不是一个实际的数据类型,指代任意一种数据类型,例如int float double等

顺序表的插入
//在顺序表L的第i个位置插入元素e
bool ListInsert(SeqList &L,int i,ElemType){if(i<1||i>L.length+1)						//判断i的范围是否在可插入范围内return flase;							//不在范围内插入失败if(L.length>=MaxSize)						//判断当前空间是否满了,满了则不能插入return flase;							//空间已满,插入失败for(int j = L.length ; j>=i ;j--){			//将插入位置之后的元素后移L.data[j]=L.data[j-i];}L.data[i-1]=e;								//在i的位置插入元素eL.length++;									//顺序表长度+1return true;插入成功
}

时间复杂度分析,时间复杂度一般关注最深层的代码,则为for循环中的移位代码。

  • 最好情况 元素插入到顺序表尾部,不需要移位则时间复杂度为O(1)
  • 最坏情况 元素插入到顺序表头部,需要将原有n个元素向后移动一位,时间复杂度为O(n)
  • 平均情况 新元素插入到每个位置的概率相等,可得平均移位次数n/2,时间复杂度为O(n)
顺序表的删除
//将表中的第i个元素删除
bool ListDelete(SeqList &L,int i){if(i<1||i>L.length+1)							//判断i的范围是否在可插入范围内return flase;								//不在范围内,删除失败for(int j=i;j<L.length;j++){					//将删除元素后续的元素向前移L.data[j-1]=L.data[j];}L.length--;										//顺序表长度-1return true;									//删除成功
}

时间复杂度分析,时间复杂度一般关注最深层的代码,则为for循环中的移位代码。

  • 最好情况 元素删除的是顺序表尾部元素,不需要移位则时间复杂度为O(1)
  • 最坏情况 元素删除的是顺序表头部元素,需要将原有n-1个元素向前移动一位,时间复杂度为O(n)
  • 平均情况 新元素插入到每个位置的概率相等,可得平均移位次数(n-1)/2,时间复杂度为O(n)
顺序表的查找
按位查找
//按位查找-查找顺序表第i位的元素
ElemType GetElem(SeqList &L,int i){return L.data[i-1];
}

时间复杂度分析,由于顺序表的存储位置是连续的,因此按位查找可以直接索引取得,时间复杂度仅位O(1);

按值查找
//按值查找,查找顺序表中e元素所在位置
int LocateElem(SeqList &L,ElemType e){for(int i=0;i<L.length;i++){if(L.data[i]==e)return i+1;}return 0
}

时间复杂度分析,时间复杂度一般关注最深层的代码,则为for循环中的遍历代码。

  • 最好情况 第一个元素为待查找元素,不需要后续遍历则时间复杂度为O(1)
  • 最坏情况 最后一个元素为待查找元素,需要遍历n个元素,时间复杂度为O(n)
  • 平均情况 新元素插入到每个位置的概率相等,可得平均移位次数(n+1)/2,时间复杂度为O(n)
顺序表的特点
  • 便于访问,能够立刻在O(1)的时间复杂度中找到待访问的值
  • 数据集中,数据的物理位置相互贴近,且每个存储节点只存储数据
  • 加长、插入、删除等操作复杂,需要大量平移元素位置

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com