1.特点:
2.定义:
我们把链表中的第一个结点的存储位置叫做头指针,最后一个结点指针为空(NULL)。
头结点的数据域一般不存储任何信息
3.头指针与头结点的异同
头指针
头结点
C语言中可以用结构指针来描述单链表:
typedef struct Node
{
ElemType data; // 数据域
struct Node* Next; // 指针域
} Node;
typedef struct Node* LinkList;
结点由存放数据元素的数据域和存放后继结点地址的指针域组成。
例1:假设p是指向线性表第i个元素的指针,则该结点ai的数据域我们可以用p->data的值是一个数据元素,结点ai的指针域可以用p->next来表示,p->next的值是一个指针。
那么p->next指向第i+1个元素。也就是指向ai+1的指针。
例2:如果p->data = ai,那么p->next->data = ?
p->next->data = ai+1。
对于单链表实现获取第i个元素的数据的操作GetElem,在算法上相对要麻烦一些:
获得链表第i个数据的算法思路:
单链表的插入:假设存储元素e的结点为s,要实现结点p、p->next和s之间逻辑关系的变化,如图:
要将结点s插入到ai和ai+1之间,只需要让s->next和p->next的指针做一点改变。
s->next = p->next;
p->next = s;
如图:
这两句代码的顺序可不可以交换过来?
先p->next = s;
再s->next = p->next;
如果先执行p->next的话会先被覆盖为s的地址,那么s->next = p->next其实就等于s->next = s了。
所以这两句是无论如何不能弄反的。
单链表第i个数据插入结点的算法思路:
代码实现:
/* 初始条件:顺序线性表L已存在,1<=i<=ListLength(L) */
/* 操作结果:在L中第i个位置之前插入新的数据元素e,L的长度加1 */
Status ListInsert(LinkList *L, int i, ElemType e)
{
int j;
LinkList p, s;
p = *L;
j = 1;
while( p && jnext;
j++;
}
if( !p || j>i )
{
return ERROR;
}
s = (LinkList)malloc(sizeof(Node));
s->data = e;
s->next = p->next;
p->next = s;
return OK;
}
假设元素a2的结点为q,要实现结点q删除单链表的操作,其实就是将它的前继结点的指针绕过指向后继结点即可。
那我们所要做的,实际上就是一步:
可以这样:p->next = p->next->next;
也可以是:q=p->next; p->next=q->next;
单链表第i个数据删除结点的算法思路:
代码实现:
/* 初始条件:顺序线性表L已存在,1<=i<=ListLength(L) */
/* 操作结果:删除L的第i个数据元素,并用e返回其值,L的长度-1 */
Status ListDelete(LinkList *L, int i, ElemType *e)
{
int j;
LinkList p, q;
p = *L;
j = 1;
while( p->next && jnext;
++j;
}
if( !(p->next) || j>i )
{
return ERROR;
}
q = p->next;
p->next = q->next;
*e = q->data;
free(q);
return OK;
}
效率问题:
无论是单链表插入还是删除算法,它们其实都是由两个部分组成:
第一部分就是遍历查找第i个元素,
第二部分就是实现插入和删除元素。
它们的时间复杂度都是O(n)。
单链表的存储结构,它的数据可以是分散在内存各个角落的,他的增长也是动态的。对于每个链表来说,它所占用空间的大小和位置是不需要预先分配划定的,可以根据系统的情况和实际的需求即时生成。
单链表的整表创建:
创建单链表的过程是一个动态生成链表的过程,从“空表”的初始状态起,依次建立各元素结点并逐个插入链表。
单链表整表创建的算法思路如下:
①头插法建立单链表
头插法从一个空表开始,生成新结点,读取数据存放到新结点的数据域中,然后将新结点插入到当前链表的表头上,直到结束为止。
就是把新加进的元素放在表头后的第一个位置:
代码实现:
/* 头插法建立单链表示例 */
void CreateListHead(LinkList *L, int n)
{
LinkList p;
int i;
srand(time(0)); // 初始化随机数种子
*L = (LinkList)malloc(sizeof(Node));
(*L)->next = NULL;
for( i=0; i < n; i++ )
{
p = (LinkList)malloc(sizeof(Node)); // 生成新结点
p->data = rand()%100+1;
p->next = (*L)->next;
(*L)->next = p;
}
}
②尾插法建立单链表
头插法建立链表虽然算法简单,但生成的链表中结点的次序和输入的顺序相反。
我们可以把思维逆过来:把新结点都插入到最后,这种算法称之为尾插法。
代码实现:
/* 尾插法建立单链表演示 */
void CreateListTail(LinkList *L, int n)
{
LinkList p, r;
int i;
srand(time(0));
*L = (LinkList)malloc(sizeof(Node));
r = *L;
for( i=0; i < n; i++ )
{
p = (Node *)malloc(sizeof(Node));
p->data = rand()%100+1;
r->next = p;
r = p; //
}
r->next = NULL;
}
当我们不打算使用这个单链表时,我们需要把它销毁;其实也就是在内存中将它释放掉,以便于留出空间给其他程序或软件使用。
单链表整表删除的算法思路如下:
代码实现:
Status ClearList(LinkList *L)
{
LinkList p, q;
p = (*L)->next;
while(p)
{
q = p->next;
free(p);
p = q;
}
(*L)->next = NULL;
return OK;
}
这段算法代码里,常见的错误就是有同学会觉得q变量没有存在的必要,只需要在循环体内直接写free(p); p = p->next; 即可?
p是一个结点,它除了有数据域,还有指针域。当我们做free(p);时,其实是对它整个结点进行删除和内存释放的工作。而我们整表删除是需要一个个结点删除的,所以我们就需要q来记载p的下一个结点。
分别从存储分配方式、时间性能、空间性能三方面来做对比:
1.存储分配方式:
2.时间性能:
查找:
插入和删除:
3.空间性能:
结论:
总之,线性表的顺序存储结构和单链表结构各有其优缺点,不能简单的说哪个好,哪个不好,需要根据实际情况,来综合平衡采用哪种数据结构更能满足和达到需求和性能。