【数据结构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);  
}

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

相关推荐
Darling噜啦啦4 天前
列表转树算法深度解析:从 Map 到 Reduce 的两种实现,面试高频考点
数据结构·算法·面试
小小工匠5 天前
Redis - 事务机制:能实现 ACID 属性吗
数据结构·redis·性能优化·并发·持久化
玖玥拾5 天前
C/C++ 数据结构(七)栈、容器适配器
c语言·数据结构·c++··容器适配器
Qres8215 天前
算法复键——树状数组
数据结构·算法
牛油果子哥q5 天前
并查集(DSU)超精讲,路径压缩、按秩合并、万能模板、连通性判定、最小生成树与刷题实战全解
数据结构·c++·最小生成树·并查集
凌波粒5 天前
LeetCode--491.递增子序列(回溯算法)
数据结构·算法·leetcode
疯狂成瘾者5 天前
Java 集合 LinkedList 详解:链表结构、常用方法和队列使用
java·开发语言·链表
WL学习笔记5 天前
单项不带头不循环链表
数据结构·链表
小糯米6015 天前
JS 数组
数据结构·算法·排序算法
小欣加油5 天前
leetcode3612 用特殊操作处理字符串I
数据结构·c++·算法·leetcode·职场和发展