链表、头指针、头结点
2016-11-16 23:04
429 查看
转自http://blog.csdn.net/zhenyusoso/article/details/6092843
图1为线性表(ZHAO, QIAN, SUN, LI, ZHOU, WU, ZHENG, WANG)的逻辑状态。头指针指示链表中第一个结点(即第一个数据元素的存储映像)的存储位置。同时,由于最后一个数据元素没有直接后继,则线性链表中最后一个结点的指针为“空”(NULL)。
图1 线性链表的逻辑状态
由上述描述可见,单链表可由头指针来唯一确定,在C语言中可用“结构指针”来描述。
[cpp]
view plain
copy
print?
//-----线性表的单链表存储结构----- typedef struct LNode{ ElemType data; struct LNode *next; }LNode, *LinkList;
有时在单链表的第一个结点之前附设一个结点,称之为头结点。头结点的数据域可以不存储任何信息,也可以存储如线性表长度等类的附加信息,头结点的指针域存储指向第一个结点的指针(即第一个元素结点的存储位置)。如图2(a)所示,此时,单链表的头指针指向头结点。若线性表为空,则头结点的指针域为“空”,如图2(b)所示。
图2
带头结点的单链表 (a)非空表;(b)空表
循环链表是另一种形式的链式存储结构。它的特点是表中最后一个节点的指针域指向头结点,整个链表形成一个环。由此,从表中任一结点出发均可找到表中其他结点,如图3所示为单链的循环链表 。
图3 单链循环表 (a)非空表;(b)空表
循环链表的操作和线性链表基本一致,差别仅在于算法中的循环条件不是p或p->next
是否为空,而是它们是否等于头指针,但有的时候,若在循环链表中设立尾指针而不设头指针(如图4(a)所示),可使某些操作简化。例如将两个线性表合并成一个表时,仅需将一个表的尾表和另一个表的头表相接。当线性表以图2.4(a)的循环链表作存储结构时,这个操作仅需改变两个指针值即可,运算时间为O (1)。合并后的表如图4(b)所示。
图4
仅设尾指针的循环链表 (a)两个链表;(b)合并后的表
以上讨论的链式存储结构的节点中只有一个指示直接后继的指针域,由此,从某个结点出发只能顺指针往后寻查其他结点。若要寻查节点的直接前趋,则需从表头指针出发。换句话说,在单链表中,NextElem的执行时间为O(1),而PriorElem的执行时间为O(n)。为克服单链表这种单向性的缺点,可利用双向链表。顾名思义,在双向链表的结点中有两个指针域,其一指向直接后继,另一指向直接前趋。在C语言中可描述如下:
[cpp]
view plain
copy
print?
//-----线性表的双向链表存储结构----- typedef struct DuLNode{ ElemType data; struct DuLNode *prior; struct DuLNode *next; }DuLNode, *DuLinkList;
和单链的循环表类似,双向链表也可以有循环表,如图5(c)所示,链表中存有两个环,图5(b)所示为只有一个表头结点的空表。在双向链表中,若d为指向表中某一个结点的指针(即d为DuLinkList型变量),则显然有
d->next->prior=d->prior->next=d
图5 双向链表示例 (a)结点结构;(b)空的双向循环链表;(c)非空的双向循环链表
单链表中,增加头结点的目的:
转自http://blog.csdn.net/zhenyusoso/article/details/6092843
图1为线性表(ZHAO, QIAN, SUN, LI, ZHOU, WU, ZHENG, WANG)的逻辑状态。头指针指示链表中第一个结点(即第一个数据元素的存储映像)的存储位置。同时,由于最后一个数据元素没有直接后继,则线性链表中最后一个结点的指针为“空”(NULL)。
图1 线性链表的逻辑状态
由上述描述可见,单链表可由头指针来唯一确定,在C语言中可用“结构指针”来描述。
[cpp]
view plain
copy
print?
//-----线性表的单链表存储结构----- typedef struct LNode{ ElemType data; struct LNode *next; }LNode, *LinkList;
//-----线性表的单链表存储结构----- typedef struct LNode{ ElemType data; struct LNode *next; }LNode, *LinkList;
有时在单链表的第一个结点之前附设一个结点,称之为头结点。头结点的数据域可以不存储任何信息,也可以存储如线性表长度等类的附加信息,头结点的指针域存储指向第一个结点的指针(即第一个元素结点的存储位置)。如图2(a)所示,此时,单链表的头指针指向头结点。若线性表为空,则头结点的指针域为“空”,如图2(b)所示。
图2
带头结点的单链表 (a)非空表;(b)空表
循环链表是另一种形式的链式存储结构。它的特点是表中最后一个节点的指针域指向头结点,整个链表形成一个环。由此,从表中任一结点出发均可找到表中其他结点,如图3所示为单链的循环链表 。
图3 单链循环表 (a)非空表;(b)空表
循环链表的操作和线性链表基本一致,差别仅在于算法中的循环条件不是p或p->next
是否为空,而是它们是否等于头指针,但有的时候,若在循环链表中设立尾指针而不设头指针(如图4(a)所示),可使某些操作简化。例如将两个线性表合并成一个表时,仅需将一个表的尾表和另一个表的头表相接。当线性表以图2.4(a)的循环链表作存储结构时,这个操作仅需改变两个指针值即可,运算时间为O (1)。合并后的表如图4(b)所示。
图4
仅设尾指针的循环链表 (a)两个链表;(b)合并后的表
以上讨论的链式存储结构的节点中只有一个指示直接后继的指针域,由此,从某个结点出发只能顺指针往后寻查其他结点。若要寻查节点的直接前趋,则需从表头指针出发。换句话说,在单链表中,NextElem的执行时间为O(1),而PriorElem的执行时间为O(n)。为克服单链表这种单向性的缺点,可利用双向链表。顾名思义,在双向链表的结点中有两个指针域,其一指向直接后继,另一指向直接前趋。在C语言中可描述如下:
[cpp]
view plain
copy
print?
//-----线性表的双向链表存储结构----- typedef struct DuLNode{ ElemType data; struct DuLNode *prior; struct DuLNode *next; }DuLNode, *DuLinkList;
//-----线性表的双向链表存储结构----- typedef struct DuLNode{ ElemType data; struct DuLNode *prior; struct DuLNode *next; }DuLNode, *DuLinkList;
和单链的循环表类似,双向链表也可以有循环表,如图5(c)所示,链表中存有两个环,图5(b)所示为只有一个表头结点的空表。在双向链表中,若d为指向表中某一个结点的指针(即d为DuLinkList型变量),则显然有
d->next->prior=d->prior->next=d
图5 双向链表示例 (a)结点结构;(b)空的双向循环链表;(c)非空的双向循环链表
单链表中,增加头结点的目的:
相关文章推荐
- 链表、头指针、头结点、第一个结点的区别与联系
- 有一个单项的链表,在没有头结点的情况下,只知道有一个指向结点B的指针p,假设这个结点B不是尾结点,删除该节点B。
- 输入一棵二叉搜索树,将该二叉搜索树转换成一个排序的双向链表。要求不能创建任何新的结点,只能调整树中结点指针的指向。
- 结构之美:单链表的头结点与头指针
- 链表总结——单链表的头指针、头结点与首元结点
- 链表的倒数第K个结点--双指针
- 数据结构——链表(头指针、头结点)
- 在链表中使用头结点与尾指针
- 雅虎面试题─有两个双向循环链表A,B,知道其头指针为:pHeadA,pHeadB,请写一函数将两链表中data值相同的结点删除
- 【链表】C++快慢两个指针解决问题(分别求倒数第k个、中间以及环的入口结点)
- LinkLists 删除链表中的一个给定指针的结点 @CareerCup
- 链表、头指针、头结点
- 链表、头指针、头结点
- 链表、头指针、头结点
- 关于对象和对象指针,涉及指向链表首结点和二叉树根结点的头指针——传指针还是传指针的指针?
- 关于链表中头指针和头结点的理解【转】
- 线性表---单链表(头结点与头指针)
- 剑指Offer——链表中快行指针用法(链表中倒数第k个结点等)
- 单链表的头指针、头结点与首元结点
- 今天开始学Java 输入一个单向链表,输出该链表中倒数第k个结点,链表的倒数第1个结点为链表的尾指针。