【数据结构3】带头指针·单向链表实现

目录

一、带头指针的链表头结构

二、接口的设计与实现

[2.1 在栈上申请空间(init接口实现初始化)](#2.1 在栈上申请空间(init接口实现初始化))

[2.2 头插接口实现](#2.2 头插接口实现)

[2.3 带头指针链表任意位置插入](#2.3 带头指针链表任意位置插入)

[2.4 带头指针链表删除元素](#2.4 带头指针链表删除元素)

[2.5 带头指针链表遍历](#2.5 带头指针链表遍历)

[2.6 销毁元素](#2.6 销毁元素)

三、test测试例


一、带头指针的链表头结构

复制代码
C

// 链表头结构
typedef struct
{
    Node_t *header;
    int count;
}ChainList_t;

二、接口的设计与实现

复制代码
C

void initChainList(ChainList_t *table);

void destroyChianList(ChainList_t *table);   

int insertChainLiHeader(ChainList_t *table,Element_t val);
int insertChainListPos(ChainList_t *table,int pos,Element_t val);

int deleteChainListElement(ChainList_t *table,Element_t val);

void showChianList(const ChainList_t *table);
2.1 在栈上申请空间(init接口实现初始化)
复制代码
C

void initChainList(ChainList_t* table)
{
    table->count = 0;
    table->header = NULL;
}
2.2 头插接口实现

带头指针的链表:引入dummy作为虚拟头结点,简化操作

复制代码
C

int insertChainLiHeader(ChainList_t* table, Element_t val)
{
    Node_t dummy;
    dummy.next = table->header;  // 指针赋值
    //下同,dummy作为虚拟头节点
    Node_t *p = &dummy;

    Node_t *new_node = malloc(sizeof(Node_t));
    if (new_node == NULL)
    {
        return -1;
    }
    // 初始化
    new_node->val = val;
    // 节点插入逻辑
    new_node->next = p->next;
    p->next = new_node;

    ++table->count;

    table->header = dummy.next;  //临时dummy存的值,最后要赋值给头指针

    return 0;
}
2.3 带头指针链表任意位置插入
复制代码
C

int insertChainListPos(ChainList_t* table,int pos, Element_t val)
{
    Node_t dummy;
    dummy.next = table->header;

    // 边界值
    if (pos < 0 || pos > table->count)
    {
        printf("insert pos invalid\n");
        return -1;
    }
    
    Node_t *p = &dummy;
    int index = -1;
    while (index < pos-1)  // index == pos-2、p = p->next退出while
    {
        p = p->next;
        ++index;
    }
    if (p == NULL)
    {
        return -1;
    }
    Node_t *new_node = malloc(sizeof(Node_t));
    new_node->val = val;
    new_node->next = p->next;
    p->next = new_node;

    ++table->count;

    table->header = dummy.next;

    return 0;
}
2.4 带头指针链表删除元素
复制代码
C

int deleteChainListElement(ChainList_t* table, Element_t val)
{
    Node_t dummy;
    dummy.next = table->header;

    Node_t *p = &dummy;

    while (p->next && p->next->val != val)
    {
        p = p->next;
    }
    if (p->next == NULL)
    {
        printf("not find\n");
        return -1;
    }
    Node_t *tmp = p->next;
    p->next = tmp->next;
    free(tmp);
    --table->count;

    table->header = dummy.next;

    return 0;

}
2.5 带头指针链表遍历
复制代码
C

void showChianList(const ChainList_t* table)
{
    // p指向第一个节点(比如100)
    Node_t *p = table->header;
    printf("chainlist : %d Element\n",table->count);

    while (p)
    {
        printf("%d\t",p->val);
        p = p->next;
    }
    printf("\n");
}
2.6 销毁元素

销毁该链表的元素区域,头不管(在堆上申请的空间才需要手动申请和释放)

复制代码
C

void destroyChianList(ChainList_t *table)
{
    Node_t dummy;
    dummy.next = table->header;

    Node_t *p = &dummy;
    Node_t *tmp;
    while (p->next)
    {
        tmp = p->next;
        p->next = tmp->next;
        free(tmp);
        --table->count;
    }
    printf("chainlist have : %d element",table->count);
}

三、test测试例

复制代码
C

int test02()
{
    ChainList_t table1;  //表在栈上
    initChainList(&table1); //进行初始化,而非创建新表
    for (int i=0;i<10;i++)
    {
        insertChainLiHeader(&table1,i+200);
    }
    insertChainListPos(&table1,8,7000);
    showChianList(&table1);
    deleteChainListElement(&table1,200);

    showChianList(&table1);

    destroyChianList(&table1);  
}

如果我的内容对你有帮助,请点赞,评论,收藏。创作不易,大家的支持就是我坚持下去的动力!

相关推荐
求学的小高3 分钟前
数据结构Day6(普通树、森林与二叉树的关系、哈夫曼编码、并查集)
数据结构·笔记·考研
算法鑫探32 分钟前
贪心算法(C 语言实现)及经典应用
c语言·数据结构·算法·贪心算法
HZ·湘怡1 小时前
任意位置 单链表 回归
c语言·链表
Peregrine91 小时前
数据结构 - > 双链表
c语言·数据结构·算法
qeen871 小时前
【数据结构】队列及其C语言模拟实现
c语言·数据结构·c++·学习·队列
AI人工智能+电脑小能手2 小时前
【大白话说Java面试题】【Java基础篇】第4题:LinkedList是单向链表还是双向链表
java·开发语言·数据结构·后端·链表·面试·list
故事和你9112 小时前
洛谷-数据结构1-4-图的基本应用1
开发语言·数据结构·算法·深度优先·动态规划·图论
破浪前行·吴14 小时前
数据结构概述
数据结构·学习
_日拱一卒15 小时前
LeetCode:25K个一组翻转链表
算法·leetcode·链表
小欣加油15 小时前
leetcode2078 两栋颜色不同且距离最远的房子
数据结构·c++·算法·leetcode·职场和发展