具有链接存储结构的线性表,它用一组地址任意的
存储单元存放线性表中的数据元素,逻辑上相邻的元素在物理上不要求也相邻,不能随机存取。一般用结点描述:结点(表示数据元素) =数据域(数据元素的映象) + 指针域(指示后继元素存储位置)
概念
在
链式存储结构中,存储数据结构的存储空间可以不连续,各数据结点的存储顺序与数据元素之间的逻辑关系可以不一致,而数据元素之间的逻辑关系是由指针域来确定的。链式存储方式既可以用于表示线性结构,也可用于表示非线性
结构。
一般来说,在线性表的链式存储结构中,各数据结点的存储符号是不连续的,并且各结点在存储空间中的位置关系与逻辑关系也不一致。对于线性链表,可以从头指针开始,沿各结点的指针扫描到链表中的所有结点。
实现
(1) 线性表的操作GetElem(L, i, &e)在链表中的实现:
基本操作为: 使指针p始终指向线性表中第j个数据元素
Status GetElem_L(LinkList L, int i, ElemType &e)// L为带
头结点的单链表的头指针。当线性表中存在第i个元素时,则将第i个数据元素的值赋给e并返回OK,否则返 回ERROR
{p = L->next; j = 1; // 初始化,p指向第一个结点,j为计数器
while (p && j); // 顺指针向后查找,直到p指向第i个元素或p为空
p = p->next; ++j; }
if ( !p || j>i ) return ERROR; // 第i个元素不存在
e = p->data; // 取第i个元素
return OK;
} // GetElem_L算法的时间复杂度为:O(ListLength(L))
(2) 线性表的操作ListInsert(&L, i, e)在链表中的实现:
基本操作为: 找到线性表中第i-1个结点,修改其指向后继的指针
Status ListInsert_L(LinkList L, int i, ElemType e) {
// 在带头结点的
单链表L中第i个数据元素之前插入数据元素e
p = L; j = 0;
while (p && j < i-1)
{ p = p->next; ++j; } // 寻找第i-1个结点
if (!p|| j > i-1) return ERROR; // i小于1或者大于表长
s = (LinkList) malloc (sizeof (LNode)); // 生成新结点
s->data = e; s->next = p->next; // 插入L中
p->next = s;
return OK;
} // LinstInsert_L算法的时间复杂度为:O(ListLength(L))
(3) 线性表的操作ListDelete(&L, i, &e)在链表中的实现:
基本操作为: 找到线性表中第i-1个结点,修改其指向后继的指针
Status ListDelete_L(LinkList L, int i, ElemType &e)
{ p = L; j = 0;} // 在带头结点的单链表L中,删除第i个元素,并由e返回其值
while (p->next && j < i-1)
{p = p->next; ++j; // 寻找第i个结点,并令p指向其前趋}
if (!(p->next) || j > i-1) return ERROR; // 删除位置不合理
q = p->next; p->next = q->next; // 删除并释放结点
e = q->data; free(q);
return OK;
} // ListDelete_L算法的时间复杂度为:O(ListLength(L))
单链线性表
在线性表的链接存储中,为了方便在表头插入和删除结点的操作,经常在表头结点(存储第一个元素的结点)的前面增加一个结点,称之为头结点或表头附加结点。这样原来的表头指针由指向第一个元素的结点改为指向头结点,头结点的数据域为空,头结点的指针域指向第一个元素的结点。
定义一个带头结点的线性链表类型:
创建带头结点的单链线性表: