如何使用宏求结构体的内存偏移地址

#define OffSet(type,field) ((size_t)&(((type*)0)->field))

在C语言中,ANSI C标准允许值为0的常量被强制转换成任何一种类型的指针,而且转换结果是一个空指针,即NULL指针,因此对0取指针的操作((type*)0)的结果就是一个类型为type*的NULL指针。 但是如果利用这个NULL指针来访问type类型的成员当然是非法的。

因为&(((type*)0)->field)的意图只不过是计算field字段的地址,C语言编译器根本不生成访问type成员的代码,而仅仅是根据type的内容布局和结构体实例地址在编译期计算这个常量地址,这样就完全避免了通过NULL指针访问内存可能出现的问题。同时又因为结构体地址是0,所以字段地址的值就是字段相对于结构体基址的偏移。

程序示例如下:

#include <stdio.h>
#define OffSet(type,field) ((size_t)&(((type*)0)->field))
struct Mystr{
	char a;
	short b;
	double c;
	int d;
};
int main()
{
	printf("%d\n",OffSet(struct Mystr,a));
	printf("%d\n",OffSet(struct Mystr,b));
	printf("%d\n",OffSet(struct Mystr,c));
	printf("%d\n",OffSet(struct Mystr,d));
	return 0;
}
程序运行结果为(32位机器,char占一个字节,short占两个字节,double占8个字节,int占4个字节):

0

2

8

16

结果分析:

上述方法避免了实例化一个type对象,而且求值是在编译期间进行,没有运行期负担,程序效率大大提高。

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: `container_of()` 是一个用于在一个数据结构中获取指向包含该结构的对象的指针的宏。它通常在内核开发中使用,特别是在实现链表或其他数据结构时使用。 在实现链表等数据结构时,通常会将数据结构嵌入到包含它的结构中。这种情况下,要获取指向包含它的结构的指针,可以使用 `container_of()` 宏。 例如,假设我们有一个包含一个 `struct list_head` 的结构体 `my_struct`: ``` struct my_struct { int data; struct list_head list; }; ``` 并且我们有一个指向 `list` 的指针 `struct list_head *ptr`,我们可以使用 `container_of()` 宏来获取指向 `my_struct` 的指针: ``` struct my_struct *p = container_of(ptr, struct my_struct, list); ``` 这个宏的实现非常巧妙,在实际使用时需要注意一些细节,比如数据结构中包含的成员变量名必须是 `list`,否则会编译错误。 ### 回答2: container_of宏是Linux内核中提供的一个非常重要的宏,并且在其他操作系统的内核开发中也有类似的实现。它的作用是通过指向结构体成员的指针来获取整个结构体的起始地址。 使用container_of宏可以方便地在提供结构体成员指针的情况下,快速地获取整个结构体的地址,从而可以方便地在后续的代码中访问结构体的其他成员。 container_of宏的使用场景非常广泛,特别在内核开发中经常使用。下面举几个常见的例子: 1. 链表操作中,可以通过链表节点的指针获取对应的结构体地址,进而进行各种链表操作。比如,在遍历链表时,可以通过container_of宏获取当前节点所在的结构体地址,并进行特定的操作。 2. 内核或驱动模块中,通过传递成员指针来获取整个包含该成员的结构体地址。在内核中,经常使用结构体指针作为参数传递,container_of宏可以方便地获取结构体的起始地址,并对整个结构体进行操作。 3. 内存分配中,container_of宏可以用于获取包含某个内存块的结构体地址。比如,在使用kmalloc分配内存时,可以通过container_of宏获取对应结构体的地址,并对其进行初始化。 总之,container_of宏的作用及使用场景是为了在操作系统内核或驱动开发中,通过成员指针快速获取对应结构体的起始地址,从而方便地进行后续的操作。它的使用可以简化代码逻辑,提高开发效率,因此被广泛使用。 ### 回答3: container_of宏是Linux内核中的一个常用宏,它的作用是通过一个结构体成员变量的指针,推导出该结构体变量的起始地址。 在C语言中,结构体变量的不同成员在内存中是依次存放的。当我们拥有一个结构体成员变量的指针时,如果想获取这个指针所属的结构体变量的起始地址,就可以使用container_of宏。 container_of宏的定义如下: #define container_of(ptr, type, member) ({ \ const typeof(((type *)0)->member) *__mptr = (ptr); \ (type *)((char *)__mptr - offsetof(type, member)); }) 其中,ptr表示待推导的结构体成员变量的指针,type表示目标结构体类型,member表示成员变量名。 使用container_of宏时,我们需要提供三个参数:成员变量的指针,结构体类型和成员变量名。宏内部使用typeof关键字来获取成员变量的指针的类型,然后使用offsetof宏来获取成员变量在结构体中的偏移量。最后,通过将成员变量的指针减去偏移量,就可以得到结构体变量的起始地址。 container_of宏在Linux内核中广泛使用,在遍历链表的时候特别有用。由于链表的节点通常嵌入在结构体中作为成员变量,当我们拥有一个链表节点的指针时,使用container_of宏可以方便地获取到包含该节点的结构体变量的起始地址。 总的来说,container_of宏是Linux内核中的一个实用工具,通过结构体成员变量的指针,可以倒推结构体变量的起始地址。这在操作链表等场景中特别有用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值