链表是⼀种物理存储结构上⾮连续、⾮顺序的存储结构,数据元素的逻辑顺序是通过链表中的
指针链接次序实现的。
结点与顺序表不同的是,链表⾥的每个结点都是独⽴申请下来的空间结点的组成主要有两个部分:当前结点要保存的数据和保存下⼀个结点的地址(指针变量)。如下图:图中指针变量 plist保存的是第⼀个结点的地址,我们称plist此时“指向”第⼀个结点,如果我们希望plist“指向”第⼆个结点时,只需要修改plist保存的内容为0x0012FFA0。链表中每个结点都是独⽴申请的(即需要插⼊数据时才去申请⼀块结点的空间),我们需要通过指针变量来保存下⼀个结点位置才能从当前结点找到下⼀个结点。
1、定义链表(结点)的结构
typedef int SLTDataType;
typedef struct SListNode {
SLTDataType data; //数据
struct SListNode* next; //指向下一个结点
}SLTNode;
2、打印结点
void SLTPrint(SLTNode* phead)
{
SLTNode* pcur = phead;
while (pcur)
{
printf("%d->", pcur->data);
pcur = pcur->next;
}
printf("NULL\n");
}
3、申请新结点
SLTNode* SLTBuyNode(SLTDataType x)
{
SLTNode* node = (SLTNode*)malloc(sizeof(SLTNode));
if (node == NULL)
{
perror("malloc fail!");
exit(1);
}
node->data = x;
node->next = NULL;
return node;
}
4、插入数据
尾插
//因为要改变结点所以传参数要传结点的地址,要用二级指针接收 void SLTPushBack(SLTNode** pphead, SLTDataType x) { assert(pphead); //申请新节点 SLTNode* newnode = SLTBuyNode(x); if (*pphead == NULL) { *pphead = newnode; } else { //找尾结点 SLTNode* pcur = *pphead; while (pcur->next) { pcur = pcur->next; } //pcur newnode pcur->next = newnode; } }
头插
void SLTPushFront(SLTNode** pphead, SLTDataType x) { assert(pphead); //头插只需要让新结点的next指向原来的头节点,再让新结点命为头节点即可 SLTNode* newnode = SLTBuyNode(x); newnode->next = *pphead; *pphead = newnode; }
5、删除
尾删
void SLTPopBack(SLTNode** pphead) { //链表为空:不可以删除 assert(pphead && *pphead); //处理只有一个结点的情况:要删除的就是头结点 if ((*pphead)->next == NULL) { free(*pphead); *pphead = NULL; } else { //找 prev ptail SLTNode* ptail = *pphead; SLTNode* prev = NULL; while (ptail->next) { prev = ptail; ptail = ptail->next; } prev->next = NULL; free(ptail); ptail = NULL; } }
头删
void SLTPopFront(SLTNode** pphead) { assert(pphead && *pphead); SLTNode* next = (*pphead)->next; //*pphead --> next free(*pphead); *pphead = next; }
6、查找
SLTNode* SLTFind(SLTNode* phead, SLTDataType x)
{
assert(phead);
SLTNode* pcur = phead;
while (pcur)
{
if (pcur->data == x)
{
//找到了返回地址
return pcur;
}
pcur = pcur->next;
}
//没有找到
return NULL;
}
7、在指定位置之前插入数据
void SLTInsert(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{
assert(pphead);
assert(pos);
if (pos == *pphead)
{
SLTPushFront(pphead, x);
}
else
{
SLTNode* newnode = SLTBuyNode(x);
//找prev :pos的前一个结点
SLTNode* prev = *pphead;
while (prev->next != pos)
{
prev = prev->next;
}
//prev newnode --> pos
newnode->next = pos;
prev->next = newnode;
}
}
8、在指定位置之后插入数据
void SLTInsertAfter(SLTNode* pos, SLTDataType x)
{
assert(pos);
SLTNode* newnode = SLTBuyNode(x);
//pos newnode --> pos->next
newnode->next = pos->next;
pos->next = newnode;
}
9、 删除pos结点
void SLTErase(SLTNode** pphead, SLTNode* pos)
{
assert(pphead && *pphead);
assert(pos);
//头删
if (pos == *pphead)
{
SLTPopFront(pphead);
}
else
{
SLTNode* prev = *pphead;
while (prev->next != pos)
{
prev = prev->next;
}
//prev pos pos->next
prev->next = pos->next;
free(pos);
pos = NULL;
}
}
10、删除pos之后的结点
void SLTEraseAfter(SLTNode* pos)
{
assert(pos && pos->next);
//pos pos->next pos->next->next
SLTNode* del = pos->next;
pos->next = pos->next->next;
free(del);
del = NULL;
}
11、销毁链表
void SListDestroy(SLTNode** pphead)
{
assert(pphead && *pphead);
SLTNode* pcur = *pphead;
while (pcur)
{
SLTNode* next = pcur->next;
free(pcur);
pcur = next;
}
*pphead = NULL;
}