(十)Linux内核中的常用宏container_of
Container_of在Linux内核中是一个常用的宏,用于从包含在某个结构中的指针获得结构本身的指针,通俗地讲就是通过结构体变量中某个成员的首地址进而获得整个结构体变量的首地址。
Container_of的定义如下:
#define OffsetOf(type, member) ((unsigned long) &(((type *)0)->member))
#define container_of(p, type, member) ((type *) ((char *)(p) - OffsetOf(type, member)))
1、其实它的语法很简单,只是一些指针的灵活应用,它分两步:
第一步,首先定义一个临时的数据类型(通过(type, member) ((unsigned long) &(((type *)0)->member))获得)member在结构体里的相对偏移地址。
第二步,用 ((type *) ((char *)(p)减去member在结构体中的偏移量,得到的值就是整个结构体变量的首地址(整个宏的返回值就是这个首地址)。
其中的语法难点就是如何得出成员相对结构体的偏移量?
2、通过例子说明,如下:
#include <stdio.h>
#define OffsetOf(type, member) ((unsigned long) &(((type *)0)->member))
typedef struct
{
int num;
char ch;
float fl;
}test_struct; int main(void)
{
printf("offsetof(test_struct, num) = %d\n",
OffsetOf(test_struct, num)); printf("offsetof(test_struct, ch) = %d\n",
OffsetOf(test_struct, ch)); printf("offsetof(test_struct, fl) = %d\n",
OffsetOf(test_struct, fl)); return ;
}
例子输出结果:

其中代码难以理解的地方就是它灵活地运用了0地址。如果觉得&(((type *)0)->member)这样的代码不好理解,那么我们可以假设在0地址分配了一个结构体变量test_struct a,然后定义结构体指针变量p并指向a(test_struct *p = &a),如此我们就可以通过&p->member获得成员member的地址。由于a的首地址为0x0,所以成员member的首地址为0x4。

最后通过强制类型转换(unsigned long)把一个地址值转换为一个整数。
3、 分析完container_of的定义,接下来举个例子来体会一下它的使用方法。
例子,如下:
#include <stdio.h>
#define OffsetOf(type, member) ((unsigned long) &(((type *)0)->member))
#define container_of(p, type, member) ((type *) ((char *)(p) - OffsetOf(type, member))) typedef struct
{
int num;
char ch;
float fl;
}test_struct; int main(void)
{
test_struct init_test_struct = { , 'C', 59.12 }; char *char_ptr = &init_test_struct.ch; test_struct *structs = container_of(char_ptr, test_struct, ch); printf(" test_struct->num = %d\n test_struct->ch = %c\n test_struct->fl = %f\n",
structs->num, structs->ch, structs->fl); return ;
}
输出结果:

由此我们知道:#define container_of(p, type, member) ((type *) ((char *)(p) - OffsetOf(type, member))) ,我们知道结构体type的地址为:结构体type的成员member的地址减去member的相对地址。这里(type *) ((char *)(p)指的是结构体type成员member的地址。
(十)Linux内核中的常用宏container_of的更多相关文章
- Linux内核中的常用宏container_of
Container_of在Linux内核中是一个常用的宏,用于从包含在某个结构中的指针获得结构本身的指针,通俗地讲就是通过结构体变量中某个成员的首地址进而获得整个结构体变量的首地址. Containe ...
- Linux内核中的常用宏container_of其实很简单【转】
转自:http://blog.csdn.net/npy_lp/article/details/7010752 开发平台:Ubuntu11.04 编 译器:gcc version 4.5.2 (Ubun ...
- Linux内核中的常用宏container_of其实很简单
http://blog.csdn.net/npy_lp/article/details/7010752 通过一个结构体变量的地址,求该结构体的首地址. #ifndef CONTAINER_OF #de ...
- 《C预处理》Linux内核中可变参数宏的用法
http://blog.csdn.net/tankai19880619/article/details/12015305
- linux内核中的宏ffs(x)
linux内核中ffs(x)宏是平台相关的宏,在arm平台,该宏定义在 arch/arm/include/asm/bitops.h #define ffs(x) ({ unsigned long __ ...
- Linux内核中双向链表的经典实现
概要 前面一章"介绍双向链表并给出了C/C++/Java三种实现",本章继续对双向链表进行探讨,介绍的内容是Linux内核中双向链表的经典实现和用法.其中,也会涉及到Linux内核 ...
- Linux 内核中的 GCC 特性
https://www.ibm.com/developerworks/cn/linux/l-gcc-hacks/ GCC 和 Linux 是出色的组合.尽管它们是独立的软件,但是 Linux 完全依靠 ...
- 剖析linux内核中的宏---------container_of
#define container_of(ptr, type, member) ({ \ const typeof(((type *)0)->member) * __mptr = (ptr); ...
- Linux内核中常用的数据结构和算法(转)
知乎链接:https://zhuanlan.zhihu.com/p/58087261 Linux内核代码中广泛使用了数据结构和算法,其中最常用的两个是链表和红黑树. 链表 Linux内核代码大量使用了 ...
随机推荐
- js认清this的第一步
学习 this 的第一步是明白 this 既不指向函数自身也不指向函数的词法作用域, 你也许被这样的解释误导过, 但其实它们都是错误的.this 实际上是在函数被调用时发生的绑定, 它指向什么完全取决 ...
- js中字符串转换为数字的方法
parseInt; parseFload; +; parseInt() 和 parseFloat() 函数会尝试逐个解析字符串中的字符,直到遇上一个无法被解析成数字的字符,然后返回该字符前所有数字字符 ...
- java面试每日一题5
题目:企业发放的奖金根据利润提成.利润(I)低于或等于10万元时,奖金可提10%:利润高于10万元,低于20万元时,低于10万元的部分按10%提 成,高于10万元的部分,可可提成7.5%:20万到40 ...
- Communication System(dp)
Communication System Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 25006 Accepted: 8925 ...
- 在opencv中实现中文输出
http://pan.baidu.com/s/1hrQTWDe 已经成功 ; 来自为知笔记(Wiz)
- Poj(1511),SPFA
题目链接:http://poj.org/problem?id=1511 嗯,最后一次写SPFA了,以后就套模板了. 题意:给出n个点和n条有向边,求所有点到源点1的来回最短路之和(保证每个点都可以往返 ...
- reactor模式学习
一.介绍reactor模式 二.使用reactor模式 三.参考 http://blog.csdn.net/swordmanwk/article/details/6170995 该文章,简单介绍了r ...
- Windows2003中IIS配置MVC提示HTTP错误 404-文件或目录未找到
Windows2003中IIS提示HTTP错误 404-文件或目录未找到 一.开启服务 启动服务“ASP.NET 状态服务”(ASP.NET State Service ),设置成“自动”. 设置完成 ...
- httpclient请求方法
/** * httpclient请求方法 * @param url 请求地址 * @param paramMap 请求参数 * @param ent 编码格式 gbk.utf-8 * @return ...
- P2680 运输计划
http://www.luogu.org/problem/show?pid=2680#sub 题目背景 公元 2044 年,人类进入了宇宙纪元. 题目描述 L 国有 n 个星球,还有 n-1 条双向航 ...