博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
C 侵入式数据结构
阅读量:6675 次
发布时间:2019-06-25

本文共 2289 字,大约阅读时间需要 7 分钟。

hot3.png

之前在网上了解到 Linux 内核开发时用的是侵入式(intrusive)数据结构,就想了解下。然后读了这篇介绍 Linux 内核中用到的双向链表实现的 。

看完那篇博客后,就想自己写个小例子感受下。

  • 实现一个简单的单向链表。
  • 然后模拟一个游戏的背包实现。添加道具到背包,然后遍历背包中的道具,最后销毁道具。
  • 我在 mingw 上测试的,若是在 Linux 上编译不过去,可以试试 -std 选项。
/** * 采用侵入式数据结构,实现一个简单的单向链表。 * gcc -o test test.c */#include 
#include
//// 链表的实现 struct list_head { struct list_head *next;};#define LIST_HEAD_INIT(name) { &(name) }#define LIST_HEAD(name) \ struct list_head name = LIST_HEAD_INIT(name)static inline void INIT_LIST_HEAD(struct list_head *list) { list->next = list;}static inline void list_add(struct list_head *new, struct list_head *head) { new->next = head->next; head->next = new;}// 对外接口// #define container_of(ptr, type, member) ({ const typeof( ((type *)0)->member ) *__mptr = (ptr); (type *)( (char *)__mptr - offsetof(type,member) );}) // 采用 typeof 编译不过#define list_entry(ptr, type, member) container_of(ptr, type, member)#define list_for_each(pos, head) for (pos = (head)->next; pos != (head); pos = pos->next)// 内部接口#define container_of(ptr, type, member) ({ \ const __auto_type __mptr = (ptr); \ (type *)( (char *)__mptr - offsetof(type,member) );})#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)//// 游戏中背包的道具。struct item_info { struct list_head list; int id;};static LIST_HEAD(items);// 向背包中添加一个道具。void additem(int id) { struct item_info *item = (struct item_info*)malloc(sizeof(struct item_info)); INIT_LIST_HEAD(&item->list); item->id = id; list_add(&item->list, &items);}int main(int argc, char *argv[]) { int count = 0; additem(200); additem(300); additem(0); struct item_info *nouse_item = NULL; struct list_head *pos = NULL, *nouse = NULL; list_for_each(pos, &items) { if (nouse) { // 要在本循环销毁上一个循环获取的道具 nouse_item = list_entry(nouse, struct item_info, list); printf("\nfree one item:%d\n", nouse_item->id); free(nouse_item); } count++; struct item_info *info = list_entry(pos, struct item_info, list); printf("id:%d ", info->id); nouse = pos; } if (nouse) { nouse_item = list_entry(nouse, struct item_info, list); printf("\nfree the last item:%d\n", nouse_item->id); free(nouse_item); } printf("total count:%d\n", count); return 0;}

思考

  • 侵入式链表中的节点并未包含具体的数据,又是如何获取到数据的呢?
  • offsetof 宏获取的偏移值,是不是在编译时就计算出来的呢,还是允许时动态算出来的。

参考

转载于:https://my.oschina.net/iirecord/blog/675980

你可能感兴趣的文章
android的快速开发框架集合
查看>>
yaffs2物理存储
查看>>
Spring入门导读——IoC和AOP
查看>>
iSCSI存储系统知识
查看>>
一步一步学ROP之linux_x64篇
查看>>
Kali linux 2016.2(Rolling)里的应用更新和配置额外安全工具
查看>>
js 实现图片实时预览
查看>>
Java 8 Optional类深度解析
查看>>
联想还是那个联想吗?
查看>>
com.panie 项目开发随笔_前后端框架考虑(2016.12.8)
查看>>
BZOJ 3529: [Sdoi2014]数表 [莫比乌斯反演 树状数组]
查看>>
ubuntu12.04中shell脚本无法使用source的原因及解决方法
查看>>
备忘录模式
查看>>
git 如何更改某个提交内容/如何把当前改动追加到某次commit上? git rebase
查看>>
eclipse里将java工程改web工程
查看>>
amazon redshift 分析型数据库特点——本质还是列存储
查看>>
C#编程(二十四)----------修饰符
查看>>
[内核]procfs和sysfs
查看>>
R语言中的数据处理包dplyr、tidyr笔记
查看>>
CSS3去除手机浏览器button点击出现的高亮框
查看>>