当前位置: 首页 > news >正文

C语言链表的操作

初学

初学C语言时,对于链表节点的定义一般是这样的:

typedef struct node {int data;struct node *next;
} Node;

向链表中添加节点:

void addNode(Node **head, int data) {Node *newNode = (Node*)malloc(sizeof(Node));newNode->data = data;newNode->next = *head;*head = newNode;
}

 删除节点和遍历节点,就不列举了

进阶

但是到了工作中,对链表的操作一般不这么玩

节点定义

工作中链表节点的定义:

struct dlist {struct dlist *next;struct dlist *prev;
};

也就是说是一个双向链表

  • next 指向下一个节点
  • prev 指向上一个节点
  • 由于是 双向循环链表,最后一个节点的 next 指向头节点,头节点的 prev 指向最后一个节点

然后我们在实际使用时,一般定义一个包含struct dlist结构的自定义结构,例如

struct my_struct {int data;struct dlist list;
};

初始化

在使用的时候,一般先初始化

static inline void dlist_init(struct dlist *list)
{list->next = list;list->prev = list;
}//调用这个接口,传入节点的地址即可
struct my_struct test;
dlist_init(&test.list);

添加节点

添加节点一般有两个接口:dlist_add和dlist_add_tail

static inline void __dlist_add(struct dlist *entry, struct dlist *prev, struct dlist *next)
{entry->next = next;entry->prev = prev;next->prev = entry;prev->next = entry;
}/* Add a new entry to the head of list */
static inline void dlist_add(struct dlist *list, struct dlist *entry)
{__dlist_add(entry, list, list->next);
}/* Add a new entry to the tail of list */
static inline void dlist_add_tail(struct dlist *list, struct dlist *entry)

删除节点

删除节点调用dlist_del接口

static inline void __dlist_del(struct dlist *entry)
{entry->next->prev = entry->prev;entry->prev->next = entry->next;
}/* Delete an entry from list */
static inline void dlist_del(struct dlist *entry)
{__dlist_del(entry);entry->next = NULL;entry->prev = NULL;
}

遍历节点

遍历节点有dlist_for_each和dlist_for_each_entry,两者的区别参考:

linux之list_for_each和list_for_each_entry函数 - 裸睡的猪 - 博客园

dlist_for_each
dlist_for_each_entry

还有一个是安全遍历,在删除节点的时候使用不使用这个接口的话,不能直接对链表进行操作,只能读取

dlist_for_each_entry_safe

通常我们要自定义一个结构体,然后结构体中包含 struct list_head 。list_head 本质上就是链表中的一个节点,而且是一个 通用节点结构,用来把自己的结构体组织成链表。可以把 struct list_head 理解为“钩子”,用来将节点一个个挂到链表上

参考:

Linux内核链表_linux的链表-CSDN博客

http://www.xdnf.cn/news/510211.html

相关文章:

  • 数字人技术的核心:AI与动作捕捉的双引擎驱动(210)
  • defer关键字:延迟调用机制-《Go语言实战指南》
  • 8.1UDP点对点聊天小项目
  • 软件架构之--论微服务的开发方法1
  • 软件工程各种图总结
  • 数据库MySQL基础2
  • 【回溯 剪支 状态压缩】# P10419 [蓝桥杯 2023 国 A] 01 游戏|普及+
  • Java大厂面试:从Web框架到微服务技术的场景化提问与解析
  • FAST-DDS源码分析PDP(一)
  • NoSQL实战指南:MongoDB与Redis企业级开发实战
  • Vue 3 动态 ref 的使用方式(表格)
  • 【Linux高级全栈开发】2.1.3 http服务器的实现
  • AI:NLP 情感分析
  • Filament引擎(一) ——渲染框架设计
  • 中级网络工程师知识点7
  • 课外活动:需了解的海象运算符(:=)
  • HTTPS的工作过程
  • 低延迟与高性能的技术优势解析:SmartPlayer VS VLC Media Player
  • 贪心、分治和回溯算法
  • 当AI自我纠错:一个简单的“Wait“提示如何让模型思考更深、推理更强
  • MySQL(21)如何查询表中的所有数据?
  • ffmpeg -vf subtitles添加字幕绝对路径问题的解决方法
  • 吴恩达机器学习(1)——机器学习算法分类
  • NetApp FAS存储系统的加密Encrytpion解决方案介绍
  • 西门子1200/1500博图(TIA Portal)寻址方式详解
  • 从零开始实现大语言模型(十五):并行计算与分布式机器学习
  • 【深度学习基础】从感知机到多层神经网络:模型原理、结构与计算过程全解析
  • java中sleep()和wait()暂停线程的区别
  • 算法题(149):矩阵消除游戏
  • 计算机系统---TPU(张量处理单元)