Chinaunix首页 | 论坛 | 博客
  • 博客访问: 2163310
  • 博文数量: 361
  • 博客积分: 10828
  • 博客等级: 上将
  • 技术积分: 4161
  • 用 户 组: 普通用户
  • 注册时间: 2010-01-20 14:34
文章分类

全部博文(361)

文章存档

2011年(132)

2010年(229)

分类: C/C++

2010-02-23 09:51:09

问: #define   offsetof(struct_t,member)   ((size_t)&((struct_t   *)0)->member)  
 
答:(struct_t   *)0是一个指向struct_t类型的指针,其指针值为   0,所以其作用就是把从地址   0   开始的存储空间映射为一个   struct_t   类型的对象。((struct_t   *)0)->member   是访问类型中的成员   member,相应地   &((struct_t   *)0)->member)   就是返回这个成员的地址。由于对象的起始地址为   0,所以成员的地址其实就是相对于对象首地址的成员的偏移地址。然后在通过类型转换,转换为   size_t   类型(size_t一般是无符号整数)。  
   
  所以,offsetoff(struct_t,member)宏的作用就是获得成员member在类型struct_t中的偏移量。
====================================================================================================
1offsetof(s,m)解析 offsetof(s,m)解析
2
3今天看代码时,发现一个有用的东东,offsetof(s,m),这是一个宏,定义如下:
4
5 #define offsetof(s,m) (size_t)&(((s *)0)->m)
6
7 然后到网上查了一下,发现还真的是很有用,附带一位大侠的解说:
8
9 struct   AAA  
10 {  
11 int   i;  
12 int   j;  
13  };  
14
15 struct AAA *pAAA;  
16  pAAA=new AAA;  
17  这时,pAAA实际上是一个Pointer, 指向某一确定的内存地址,比如0x1234;  
18  而 pAAA->i 整体是一个int型变量,其地址是&(pAAA->i) ,'&'为取址运算符;  
19  那么&(pAAA->i)一定等于0x1234,因为i是结构体AAA的第一个元素。  
20  而&(pAAA->j)一定是0x1234 + 0x4 = 0x1238; 因为sizeof(int) = 4;
21
22  这个做法的巧妙之处就是:它把“0”作为上例中的pAAA,那么 &(pAAA->j)就是j的  
23  offset啦
24
25  解析结果是:  
26  (s   *)0 ,将 0 强制转换为Pointer to   "s"
27  可以记 pS = (s *)0 ,pS是指向s的指针,它的值是0;  
28  那么pS->m就是m这个元素了,而&(pS->m)就是m的地址,而在本例中就是offset啦  
29
30  再把结果强制转换为size_t型的就OK 了,size_t其实也就是int啦!!
阅读(898) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~