链表基础2——带头结点单向非循环链表的基本操作(c语言实现)

目录

头结点

带头结点单向不循环链表

链表的基本操作 

一,链表的初始化

二,创建新结点

三,指定结点的后插操作

四,指定结点的前插操作

五,在链表头部插入新结点

六,在链表尾部插入新结点 

七,删除链表的第一个结点 

八,删除链表的最后一个结点

九,释放链表

十,打印链表

完整代码


头结点

头结点是数据结构中,在单链表的第一个结点之前附设的一个结点。

它没有直接前驱,其数据域可以不存储任何信息,也可以存储如链表长度等附加信息。

头结点的指针域存储指向第一个结点的指针,即第一个元素结点的存储位置。

头结点的作用是使所有链表(包括空表)的头指针非空,并使对单链表的插入、删除操作不需要区分是否为空表或是否在第一个位置进行,从而与其他位置的插入、删除操作一致。在链表中设置头结点还可以方便地判断链表是否为空:当头结点的指针域为空时,表示链表为空。

然而,头结点并不是链表所必需的,它主要是为了操作的统一和方便而设立的。无论是否有头结点,头指针始终指向链表的第一个结点。如果有头结点,头指针就指向头结点;如果没有头结点,头指针则指向链表的第一个数据结点。

带头结点单向不循环链表

我们也是借用结构体来表示一个单链表的定义

typedef int SLTDatatype;

typedef struct SListNode
{
SLTDataType data;
struct SListNode*next;
}SLTNode;

 SLTNode是“Singly Linked List Node”的缩写,它表示单链表中的结点。

在“SLTNode”这个缩写中,中间的“T”通常代表“Type”或者“Node”的缩写。在这里,“T”更是为了强调这是一个特定的类型(Type)或者是一个结点(Node)的表示。

链表的基本操作 

带头结点单向不循环链表的基本操作包括:

  1. 创建链表:首先需要定义一个链表结构体,其中每个结点包含一个数据域和一个指向下一个结点的指针。然后,通过动态内存分配(如使用malloc函数)来创建链表的各个结点,并将它们按照顺序连接起来。头结点作为链表的起始点,其指针域指向第一个实际的数据结点。
  2. 清空链表:遍历链表,逐个释放每个结点的内存空间,直到链表为空。注意,需要确保正确处理头结点,避免内存泄漏。
  3. 销毁链表:在清空链表后,还需要释放头结点的内存空间,以完全销毁整个链表。
  4. 头插法:在链表的头部插入新结点。具体操作为:创建一个新结点,将其数据域设置为要插入的数据,然后将其指针域指向头结点的下一个结点,最后更新头结点的指针域,使其指向新结点。
  5. 尾插法:在链表的尾部插入新结点。这需要遍历链表找到最后一个结点,然后创建一个新结点,将其数据域设置为要插入的数据,并将最后一个结点的指针域指向新结点。
  6. 任意位置插入法:在链表的任意位置插入新结点。首先找到要插入位置的前一个结点,然后创建一个新结点,将其数据域设置为要插入的数据,并将其指针域指向要插入位置的原结点,最后更新前一个结点的指针域,使其指向新结点。
  7. 头删法:删除链表的头部结点。具体操作为:将头结点的下一个结点作为新的头结点,然后释放原头结点的内存空间。
  8. 尾删法:删除链表的尾部结点。这需要遍历链表找到倒数第二个结点,然后将其指针域设置为NULL,并释放原尾部结点的内存空间。
  9. 任意位置删除法:删除链表的任意位置结点。首先找到要删除结点的前一个结点,然后更新前一个结点的指针域,使其跳过要删除的结点,并指向要删除结点的下一个结点,最后释放要删除结点的内存空间。
  10. 查询链表中是否有想要的数据:遍历链表,逐个比较结点的数据域与要查询的数据是否相等,若相等则返回该结点的位置或数据,否则继续遍历直到链表结束。

这些基本操作构成了带头结点单向不循环链表的基本功能,可以根据具体需求进行组合和扩展。需要注意的是,在实际编程中,还需要考虑错误处理、边界条件以及内存管理的安全性等问题。

一,链表的初始化

带头结点单向非循环链表和不带头结点单向非循环链表的初始化操作不同,带头结点点的需要先搞出头结点来,而另外一个不用

void SLTInit(SLTNode** pphead)
{
	*pphead = (SLTNode*)malloc(sizeof(SLTNode));//创建头结点
	if (*pphead == NULL)
	{
		perror("malloc fail");
		return;
	}
	(*pphead)->next = NULL;//头结点的next置空
    (*pphead) -> data = 0;//仅仅是为了防止潜在的错误而设定的
}

经过这个操作我们得注意一个点:*pphead代表的是头结点,而不是第一个元素。

第一个元素是(*pphead)->next 

二,创建新结点

SLTNode* BuySLTNode(SLTDataType x)
{
	SLTNode* newnode = (SLTNode*)malloc(sizeof(SLTNode));//分配内存
	if (newnode == NULL)//如果内存分配失败
	{
		perror("malloc fail");
		return NULL;
	}
//给新结点赋值
	newnode->data = x;
	newnode->next = NULL;
 
	return newnode;
}

三,指定结点的后插操作

//通常情况下,我们是按照*pos是头节点的标准来设计这个操作的
//在指定结点pos后面添加一个元素
// 在指定结点pos后面添加一个元素  
void SLTInsertAfter(SLTNode* pos, SLTDataType x) {  
    assert(pos != NULL && pos->next != NULL); // 确保pos不为空且pos不是链表的最后一个节点  
    SLTNode* newnode = BuySLTNode(x);  
    newnode->next = pos->next;  
    pos->next = newnode;  
}

四,指定结点的前插操作

// 在指定结点pos前添加一个元素  
void SLTInsertFront(SLTNode** pphead, SLTNode* pos, SLTDataType x) {
    assert(pphead != NULL && *pphead != NULL && pos != NULL);
    if (*pphead == pos) {
        perror("头节点不能前插\n");
        exit(-1);
    }
    SLTNode* cur = *pphead;
    while (cur->next != pos) {
        if (cur->next == NULL) {
            perror("目标结点pos不在链表中\n");
            exit(-1);
        }
        cur = cur->next;
    }
    SLTNode* newnode = BuySLTNode(x);
    newnode->next = pos;
    cur->next = newnode;
}

五,在链表头部插入新结点

// 在链表头部插入新结点  
void ListInsertFront(SLTNode** pphead, SLTDataType data) {
    assert(pphead != NULL && *pphead != NULL);
    SLTNode* newnode = BuySLTNode(data);
    newnode->next = (*pphead)->next;
    (*pphead)->next = newnode;
}

六,在链表尾部插入新结点 

// 在链表尾部插入新结点   
void InsertRear(SLTNode** pphead, SLTDataType data) {
    assert(pphead != NULL && *pphead != NULL);
    SLTNode* newnode = BuySLTNode(data);
    SLTNode* tail = *pphead;
    while (tail->next) {
        tail = tail->next;
    }
    tail->next = newnode;
}

七,删除链表的第一个结点 

// 删除链表的头部结点  
void DeleteFront(SLTNode** pphead) {
    if ((*pphead)->next == NULL) {//我们要注意*pphead是头节点,不是第一个结点
        printf("List is empty!\n");
        return;
    }
    SLTNode* p = (*pphead)->next;
    (*pphead)->next = p->next; // 头结点指向原第二个结点  
    free(p); // 释放原第一个结点内存  
}

八,删除链表的最后一个结点

// 删除链表的尾部结点  
void DeleteRear(SLTNode** pphead) {
    if ((*pphead)->next == NULL) {
        printf("List is empty!\n");
        return;
    }
    SLTNode* p = *pphead;
    SLTNode* q = NULL;
    while (p->next->next) { // 找到倒数第二个结点  
        q = p;
        p = p->next;
    }
    //p为倒数第二个
    //q为倒数第三个
    q->next = NULL; // 最后一个结点的前一个结点指向NULL  
    free(p->next); // 释放原尾部结点内存  
}

九,释放链表

// 释放链表内存  
void FreeList(SLTNode* head) {
    SLTNode* cur = head;
    while (cur) {
        SLTNode* next = cur->next;
        free(cur);
        cur = next;
    }
}

十,打印链表

// 打印链表  
void PrintList(SLTNode* head) {
    SLTNode* cur = head->next;
    while (cur) {
        printf("%d ", cur->data);
        cur = cur->next;
    }
    printf("\n");
}

完整代码

#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
typedef int SLTDataType;

typedef struct SListNode
{
	SLTDataType data;
	struct SListNode* next;
}SLTNode;

// 初始化链表  
void SLTInit(SLTNode** pphead) {
    *pphead = (SLTNode*)malloc(sizeof(SLTNode));
    if (*pphead == NULL) {
        perror("malloc fail");
        exit(EXIT_FAILURE);
    }
    (*pphead)->data = 0; // 初始化头结点的data字段  
    (*pphead)->next = NULL;
}

// 创建新结点  
SLTNode* BuySLTNode(SLTDataType x) {
    SLTNode* newnode = (SLTNode*)malloc(sizeof(SLTNode));
    if (newnode == NULL) {
        perror("malloc fail");
        exit(EXIT_FAILURE);
    }
    newnode->data = x;
    newnode->next = NULL;
    return newnode;
}

// 在指定结点pos后面添加一个元素  
void SLTInsertAfter(SLTNode* pos, SLTDataType x) {
    assert(pos != NULL && pos->next != NULL); // 确保pos不为空且pos不是链表的最后一个节点  
    SLTNode* newnode = BuySLTNode(x);
    newnode->next = pos->next;
    pos->next = newnode;
}

// 在指定结点pos前添加一个元素  
void SLTInsertFront(SLTNode** pphead, SLTNode* pos, SLTDataType x) {
    assert(pphead != NULL && *pphead != NULL && pos != NULL);
    if (*pphead == pos) {
        perror("头节点不能前插\n");
        exit(-1);
    }
    SLTNode* cur = *pphead;
    while (cur->next != pos) {
        if (cur->next == NULL) {
            perror("目标结点pos不在链表中\n");
            exit(-1);
        }
        cur = cur->next;
    }
    SLTNode* newnode = BuySLTNode(x);
    newnode->next = pos;
    cur->next = newnode;
}

// 在链表头部插入新结点  
void ListInsertFront(SLTNode** pphead, SLTDataType data) {
    assert(pphead != NULL && *pphead != NULL);
    SLTNode* newnode = BuySLTNode(data);
    newnode->next = (*pphead)->next;
    (*pphead)->next = newnode;
}

// 在链表尾部插入新结点  
void InsertRear(SLTNode** pphead, SLTDataType data) {
    assert(pphead != NULL && *pphead != NULL);
    SLTNode* newnode = BuySLTNode(data);
    SLTNode* tail = *pphead;
    while (tail->next) {
        tail = tail->next;
    }
    tail->next = newnode;
}

// 释放链表内存  
void FreeList(SLTNode* head) {
    SLTNode* cur = head;
    while (cur) {
        SLTNode* next = cur->next;
        free(cur);
        cur = next;
    }
}

// 打印链表  
void PrintList(SLTNode* head) {
    SLTNode* cur = head->next;
    while (cur) {
        printf("%d ", cur->data);
        cur = cur->next;
    }
    printf("\n");
}
// 删除链表的头部结点  
void DeleteFront(SLTNode** pphead) {
    if ((*pphead)->next == NULL) {//我们要注意*pphead是头节点,不是第一个结点
        printf("List is empty!\n");
        return;
    }
    SLTNode* p = (*pphead)->next;
    (*pphead)->next = p->next; // 头结点指向原第二个结点  
    free(p); // 释放原第一个结点内存  
}

// 删除链表的尾部结点  
void DeleteRear(SLTNode** pphead) {
    if ((*pphead)->next == NULL) {
        printf("List is empty!\n");
        return;
    }
    SLTNode* p = *pphead;
    SLTNode* q = NULL;
    while (p->next->next) { // 找到倒数第二个结点  
        q = p;
        p = p->next;
    }
    //p为倒数第二个
    //q为倒数第三个
    q->next = NULL; // 最后一个结点的前一个结点指向NULL  
    free(p->next); // 释放原尾部结点内存  
}
// 测试代码  
int main() {
    SLTNode* head = NULL;
    SLTInit(&head);

    ListInsertFront(&head, 1);
    ListInsertFront(&head, 2);
    InsertRear(&head, 3);
    SLTInsertAfter(head->next, 4); // 插入到第二个节点后  
    SLTInsertFront(&head, head->next->next, 5); // 插入到第三个节点前  

    PrintList(head);

    FreeList(head);

    return 0;
}

  • 17
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值