对offsetof、 container_of宏和结构体的理解
offsetof 宏
#include<stdio.h>
#define offsetoff(type, member) ((int)&((type*)0)->member)
/*
((type*)0)->member 释义:声明一个相应类型的结构体指针,该指针指向0地址处。再通过该指针访问各元素。我们只要获取一个指针就能访问其内部的元素吗,可以这么搞?其实我是想联系全局和非全局变量,通过上面这个代码也许你不明白我要表达的意思。请继续慢慢看,直到本文后面,结合本文后面对单链表的一个疑点及分析,你就会体会更深了。暂时这里先不管这个,
指向0地址处的结构体指针访问各元素得到的地址值就是其偏移量了? 看来这是一个公认的道理。
我不能解释为什么,但是我可以分析一下通过 '->"、 普通结构体指针访问其元素的性质,通过对比,
如果性质相同,那就认了这个道理,不必深究。
*/
typedef struct stu{
char a;
int b;
char c;
}*p1;
int main(void)
{
struct stu stu1;
stu1.b = 666;
p1 p = &stu1;
printf("&stu1 = %p.\n", &stu1);
printf("p = %p.\n", p);
printf("stu1.b = %d.\n", p->b);
printf("&(stu1.b) = %p.\n", &(p->b));
}
/*
aston@ubuntu:/mnt/hgfs/shared$ gcc a.c
aston@ubuntu:/mnt/hgfs/shared$ ./a.out
&stu1 = 0xbfc485b4.
p = 0xbfc485b4.
stu1.b = 666.
&(stu1.b) = 0xbfc485b8. //比上面多4个字节,合理。
打印出来的都是0xbfc485b + x,
如果直接先看这个例子,那我也很能理解各元素的地址打印出来将会是什么样,甚至都不会细想。想当然地认为:
指针 + 结构体嘛,考虑下对齐,就知道各元素地址了。但是为什么一看见offsetoff就有点迷糊了呢?
小结论:一个知识点只有在多种不同场景下被应用,才能加深印象。
回到主问题:我们通过实验发现,结构体指针不同时,编译器都会自动增加偏移量。
大结论: 我们通过“->”的方式访问结构体元素,其实是利用了编译器自动帮我们计算元素偏移量的特点。
只有理解了这点,才能更好地理解 offsetof 宏。
另外,这个例子我并没有产生“我们只要获取一个指针就能访问其内部的元素吗,可以这么搞”的疑惑,
因为在这个例子里,stu1.b = 666;和printf("stu1.b = %d.\n", p->b);这两句代码都在一个代码块{}中,
我p->b访问结构体元素,p 和 b 都在同一个代码块{},当然可以访问啊。
*/
再来看这个代码:
#include<stdio.h>
#include <stdlib.h>
#include <strings.h>
typedef struct node{
unsigned int data;
struct node* pNext; //难道这个指针是全局的吗?
}Node;
Node* creat_node(unsigned int data)
{
Node *p = (Node*)malloc(sizeof(Node));
printf("sizeof(Node)-8- = %d.\n", sizeof(Node));
if(NULL ==p)
{
printf("malloc error");
return NULL;
}
bzero(p,sizeof(Node));
p->data = data;
p->pNext = NULL;
return p;
}
int main(void)
{
Node* pheader= NULL;
pheader = creat_node(666);
pheader->pNext = creat_node(777);
pheader->pNext->pNext = creat_node(888);
printf("node1: data:%d\n", pheader->data);
//creat_node的返回值只是p,但是却可以使用pNext这个指针(即访问p指向节点内的元素)。难道这个指针是全局的吗?data同样作为节点内的元素,data也是全局的?
printf("node1: data:%d\n", pheader->pNext->data);
printf("node1: data:%d\n", pheader-pNext-pNext->data);
}
我们写个测试代码
局部的结构体变量aaa。
#include <stdio.h>
typedef struct node{
int data;
}Node;
void func(void)
{
Node aaa;
aaa.data = 100;
}
int main(void)
{
printf("%d\n", aaa.data);
return 0;
}
编译报错error: ‘aaa’ undeclared (first use in this function)
测试结果:不是全局的。
结论:我们只要获取了结构体的指针,我们就可以访问结构体内的元素(member).
再结合这点理解文首的((type*)0)->member。哪怕你member是局部变量,我也通过结构体指针,就能获取你。
再来证实一下:
typedef struct node{
int data;
}Node;
int func(void)
{
int addr = 0;
Node aaa;
addr = (int)&aaa;
aaa.data = 100;
return addr;
}
int main(void)
{
int a = func();
printf("%d\n", ((Node*)a)->data);
return 0;
}
container_of宏
#include <stdio.h>
struct mystruct
{
char a; // 0
int b; // 4
short c; // 8
};
// TYPE是结构体类型,MEMBER是结构体中一个元素的元素名
// 这个宏返回的是member元素相对于整个结构体变量的首地址的偏移量,类型是int
#define offsetof(TYPE, MEMBER) ((int) &((TYPE *)0)->MEMBER)
// ptr是指向结构体元素member的指针,type是结构体类型,member是结构体中一个元素的元素名
// 这个宏返回的就是指向整个结构体变量的指针,类型是(type *)
#define container_of(ptr, type, member) ({ \
const typeof(((type *)0)->member) * __mptr = (ptr); \
(type *)((char *)__mptr - offsetof(type, member)); })
//我的下面两句代码不行。为什么?????????
//#define container_of(ptr, type, member) ((type*)((char*)ptr - offsetof(type, member));)
//#define container_of(ptr, type, member) ((type*)((int)ptr - offsetof(type, member));)
int main(void)
{
struct mystruct s1;
struct mystruct *pS = NULL;
short *p = &(s1.c); // p就是指向结构体中某个member的指针
printf(" &s1 :%p.\n", &s1);
// 问题是要通过p来计算得到s1的指针
pS = container_of(p, struct mystruct, c);
printf("pS is :%p.\n", pS);
return 0;
}
对offsetof、 container_of宏和结构体的理解的更多相关文章
- typeof, offsetof, container_of宏
container_of宏实现如下: #define container_of(ptr, type, member) ({ \ )->member ) *__mptr = (ptr); \ (t ...
- 深入delphi编程理解之消息(二)发送消息函数及消息编号、消息结构体的理解
一.delphi发送消息的函数主要有以下三个: (一).SendMessage函数,其原型如下: function SendMessage( hWnd: HWND; {目标句柄} Msg: UINT; ...
- C++ 结构体指针理解
上一篇基础链接https://www.cnblogs.com/xuexidememeda/p/12283845.html 主要说一下链表里面双重指针 先说一下结构体 typedef struct LN ...
- linux中offsetof与container_of宏定义
linux内核中offsetof与container_of的宏定义 #define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->M ...
- (转)offsetof与container_of宏[总结]
1.前言 今天在看代码时,遇到offsetof和container_of两个宏,觉得很有意思,功能很强大.offsetof是用来判断结构体中成员的偏移位置,container_of宏用来根据成员的地址 ...
- offsetof与container_of宏[总结]
1.前言 今天在看代码时,遇到offsetof和container_of两个宏,觉得很有意思,功能很强大.offsetof是用来判断结构体中成员的偏移位置,container_of宏用来根据成员的地址 ...
- Linux中的两个经典宏定义:获取结构体成员地址,根据成员地址获得结构体地址;Linux中双向链表的经典实现。
倘若你查看过Linux Kernel的源码,那么你对 offsetof 和 container_of 这两个宏应该不陌生.这两个宏最初是极客写出的,后来在Linux内核中被推广使用. 1. offse ...
- offsetof与container_of宏分析
offsetof宏:结构体成员相对结构体的偏移位置 container_of:根据结构体成员的地址来获取结构体的地址 offsetof 宏 原型: #define offsetof(TYPE, MEM ...
- 结构体中使用#define定义宏
struct hostent { char *h_name; /* official name of host */ char **h_aliases; /* ...
随机推荐
- springMVC使用JSR303数据校验
JSR303注解 hibernate validate是jsr 303的一个参考实现,除支持所有的标准校验注解外,他还支持扩展注解 spring4.0拥有自己独立的数据校验框架,同时支持jsr 303 ...
- 跟我一起学.NetCore之日志作用域及第三方日志框架扩展
前言 上一节对日志的部分核心类型进行简单的剖析,相信现在再使用日志的时候,应该大概知道怎么一回事了,比如记录器是怎么来的,是如何将日志内容写入到不同目的地的等:当然还有很多细节没深入讲解,抽时间小伙伴 ...
- qqmini
QQ玩一玩最新调试方法 https://blog.csdn.net/zyw_java/article/details/83686645 LayaBox 接入QQ玩一玩 轻游戏流程 https://bl ...
- Linux常用命令--不断更新
Linux命令: !. 1.[root@loc8lhost/root]# 表示登陆进去系统,其中#是超级⽤用户也即root⽤用 户的系统提示符 #. 2.reboot命令可以重启系统 $. 3.关闭系 ...
- 华为手机logcat中不显示log.e以下级别日志的解决方法
(1) 进入拨号界面输入:*#*#2846579#*#* (2) 进入“后台设置” ——>“LOG设置” (3) 点击选择“AP日志” (4) 部分手机可能需要重启.
- 【已学完】UGUI Schedule
章节 内容 签到 Unity4.6 New UI(UGUI) 1.UGUI概述与Canvas画布介绍(一) 5月14日 2.Canvas画布介绍(二) 5月14日 3.Text控件 5月14日 4.I ...
- Linux 操作系统 基础
root: 管理员 /: 根目录[windows : 计算机] ~: 家目录: 管理员:/root/.. 非管理员:/home/... 命令提示符: [root@localhost abc]#: ro ...
- ES6特性整理
ESMAScript6.0 ES6 兼容 IE10+ .Chrome.Firefox 要想兼容IE10以下的有两种方法: 用 babel 工具 borwer.js ,在 script 标签里添加 ty ...
- MySql 实现数组根据下标获取对应值逻辑(array[i]逻辑)
在使用sql模拟一段java逻辑开发时碰到有一段逻辑为从字符串数组中根据下标获取对应的值的情况,百度了一番没有发现有类似功能的函数和现成的实现方式,经过调试弄出来了,记录下来,以备参考 //举例:从数 ...
- DevOps-实践心得
基于最近几年从事与DevOps的相关实践,对这篇文章的观点深有体会,所以记录在这里.加粗部分是我比较深有体会的,但是对于最后作者对于"运维"有些悲观,我有点不敢苟同,反而对于运维的 ...