获取struct 中的member variable 的offset的宏定义

来源:互联网 发布:算法博弈论中文版 pdf 编辑:程序博客网 时间:2024/06/05 04:36

转自:

http://www.linuxidc.com/Linux/2013-01/78004.htm


1.offset宏讲解

#defineoffsetof(TYPE, MEMBER) ((size_t) &((TYPE*)0)->MEMBER)

对这个宏的讲解我们大致可以分为以下4步进行讲解:

1>((TYPE *)0 )  将0地址当作是一个TYPE类型的实例对象的地址强制 "转换" TYPE结构类型的指针;

2>((TYPE*)0)->MEMBER 访问TYPE类型实例对象中的MEMBER数据成员;

3>&(( (TYPE *)0 )->MEMBER)取出TYPE类型实例对象中的数据成员MEMBER的地址;

4>(size_t)(&(((TYPE*)0)->MEMBER))结果转换为size_t类型。

宏offsetof的巧妙之处在于将0地址强制转换为TYPE结构类型的指针,TYPE结构以内存空间首地址0作为起始地址,则成员地址自然为偏移地址。可能有的读者会想是不是非要用0呢?当然不是,我们仅仅是为了计算的简便。也可以使用是他的值,只是算出来的结果还要再减去该数值才是偏移地址。来看看如下的代码:

 #include<stdio.h>

 #defineoffsetof(TYPE, MEMBER) ((size_t) &((TYPE *)4)->MEMBER)

  structtest_struct {

         int num;

         char ch;

         float f1;

  };

 int main(void)

  {

   printf("offsetof (struct test_struct,num)=%d\n",offsetof(structtest_struct,num)-4);

   printf("offsetof (structtest_struct,ch) =%d\n",offsetof(structtest_struct,ch)-4);

   printf("offsetof (struct test_struct,f1)=%d\n",offsetof(structtest_struct,f1)-4);

         return 0;

  }

运行结果为:

jibo@jibo-VirtualBox:~/cv_work/work/list/offset$ ./main

offsetof (structtest_struct,num) =0

offsetof (structtest_struct,ch) =4

offsetof (structtest_struct,f1) =8

为了让大家加深印象,我们在代码中没有使用0,而是使用的4,所以在最终计算出的结果部分减去了一个4才是偏移地址,当然实际使用中我们都是用的是0。

二.举例体会offsetof宏的使用:

 #include<stdio.h>

 #defineoffsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)

  structtest_struct {

         int num;

         char ch;

         float f1;

  };

 int main(void)

  {

     printf("offsetof(struct test_struct,num) =%d\n",offsetof(structtest_struct,num));

   printf("offsetof (structtest_struct,ch) =%d\n",offsetof(structtest_struct,ch));

   printf("offsetof (struct test_struct,f1)=%d\n",offsetof(structtest_struct,f1));

         return 0;

  }

执行结果为:

jibo@jibo-VirtualBox:~/cv_work/work/list/offset$ ./main

offsetof (structtest_struct,num) =0

offsetof (structtest_struct,ch) =4

offsetof (structtest_struct,f1) =8

0 0