线性表(数据结构)

线性表

1.线性表
定义:n(>=0)个数据元素的有限序列,记为L=(a1,a2,…,an)。(ai是数据元素,n是表长度,a1是首元素,an是尾元素)
特点
1.除第一个元素外,其他每一个元素都有且仅有一个直接前趋。
2.除最后一个元素外,其他每一个元素都有且仅有一个直接后继。
要点:
表中元素具有逻辑上的顺序性,在序列中各元素排列有其先后次序,有唯一的首元素和尾元素;
表中元素个数有限;
表中元素都是数据元素,即每一表元素都是原子数据,不允许表中套表。
表中元素的数据类型都相同。这意味着每一表中元素占有相同的存储空间。
2.顺序表
定义:将线性表中的元素相继存放在一个连续的存储空间中,即构成顺序表。他是线性表的顺序存储表示,可利用一位数组描述存储结构。
**特点:**元素的逻辑顺序与物理顺序一致;
可顺序存储,可按下标直接存取。
顺序表的连续存储方式:
LOC(i)=LOC(i-1)+l=a+i*l
LOC是元素存储位置,l是元素大小。
静态结构定义:

#define maxSize 100   //最大允许长度
typedef int dataType; //元素数据类型 
typedef struct{
	dataType data[maxSize];     //存储数组 
	int n;          //当前表元素个数 
}SeqList; 

顺序表静态定义,假定L是一个类型为SeqList的顺序表,一般用L.data[i]来访问他。
表一旦装满,不能扩充。
动态结构定义

#define initSize 100   //最大允许长度
typedef int dataType; //元素数据类型 
typedef struct{
	dataType *data;     //存储数组 
	int n;          //当前表元素个数  
	int maxSize;    //表的最大长度 
}SeqList; 

顺序表动态定义,他可以扩充,新的大小计入数据成员maxSize中。
顺序表基本运算实现
构造一个空的顺序表:

void initList(SeqList&L) 
{
	L.data=(dataType*)malloc(initSize*sizeof(dataType));
	if(L.data==NULL)
	{
		printf("存储分配失败!\n");
		exit(1);
	}
	L.n=0;
	L.maxSize=initSize;
}

3.链表
线性链表是线性表的链接储存表示。元素中间的逻辑顺序是通过各节点中的链接指针来表示的。
线性链表分类:
单链表
循环链表
双向链表
链表中第一个元素节点称为首元结点,最后一个节点称为尾结点。(首元结点不是头结点)
单链表
特点:每个元素(表项)由节点(node)构成。
节点可以连续,可以不连续存储。
节点的逻辑顺序与物理顺序可以不一致。
表可以扩充。
单链表结构定义:

typedef char datatype;
typedef struct node     //链表结点 
{
	datatype data;        //结点数据域 
	struct node *link;     //结点链域 
}linknode,*linklist;       //链头指针 

使用时定义实际链表,只需定义链表头指针。

linklist first; //链表头指针 

在链表中,如果没有定义重载“++”函数,不能使用p++这样的语句进行到逻辑上的下一个节点,一般用p=p->link进行到下一个节点。
循环链表
循环单链表是单链表的变形。链表尾结点的link指针不是NULL,而是指向了表的前端。请添加图片描述
为简化操作,在循环单链表中加入头结点。请添加图片描述

循环单链表的判空条件是:first->link=first。
特点:只要知道表中某一结点的地址,就可搜寻到所有其他结点的地址。在搜寻过程中,没有一个结点的link域为空。
循环单链表的所有操作的实现类似于单链表,差别在于检测到链尾,指针不为NULL,而是回到链头。
循环单链表的结构定义:

typedef int datatype;
typedef struct node   // 循环链表定义 
{
	datatype data;     //结点数据 
	struct node *link;   //后继结点指针 
}circnode,*circlist;

在链表中将指针p定位于第i个结点的操作为:

circnode *p=first; //firsr是头结点 
int k=0;
while(p->link!=first&&k<i)   //回到头结点失败 
{
	p=p->link;    //否则p指到目标 
	k++;
}

双向链表
双向链表是指在前趋和后继方向都能遍历的线性链表。双向链表每个结点的结构为:
请添加图片描述
双向链表通常采用带头结点的循环双链表形式,每一个结点处于两个链中:
请添加图片描述
结点指向:
p->llink(left)指示结点p的前趋结点;
p->rlink(right)指示结点p的后继结点;
p->llink->rlink或者p->rlink->llink指示p结点本身。
循环双链表的定义:

typedef int datatype;
typedef struct node   // 结点定义 
{
	datatype data;     //结点数据 
	int freq;          //访问计数 
	struct node *llink,*rlink;   //指针 
}dblcnode,*dbllist;     //双向链表 

单链表寻找结点后继的时间复杂度为O(1),寻找结点前趋的时间复杂度为O(n);而双向链表寻找后继和前趋的时间复杂度都是O(1)。
建立空的循环双链表:

void initdbllist(dbllist&first) 
{
	first=(dblnode*)malloc(sizeof(dblnode));
	if(first=NULL)
	{
		printf("存储分配失败!\n");
		exit(1);
	}
	first->llink=first->rlink=first;
	first->freq=0;
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值