((size_t) &((TYPE*)0)->MEMBER)
2017-04-11 15:01
260 查看
define offsetof(TYPE, MEMBER) ((size_t) &((TYPE*)0)->MEMBER)
对于这个宏可以大致分为5步:1. 0
2. ((TYPE *)0)
3. ( ((TYPE *)0)->MEMBER )
4. &( ((TYPE *)0)->MEMBER )
5. ( (size_t) &( ((TYPE *)0)->MEMBER )
1、内存地址开始于0;
2、将0转换为type类型的结构体指针,换句话说就是让编译器认为这个结构体是开始于程序段起始位置;
3、引用结构体中MEMBER成员;
4、取地址符&,我们这里不关注结构体成员的内容,只取该成员的地址;
5、将取到的地址强制转换为size_t类型。
因为这个结构体的起始地址被指定为0,所以取到的结构体成员的绝对地址(当转换为数字)就是这个成员在结构体中的偏移量。
这个代码之所以没有风险,是因为这里面没有写任何内存位置,甚至没有访问任何内存位置。只是操作了指向这些位置的指针,而指针一般存储在机器寄存器或是通常的本地堆栈。
以下代码也是正确的:
( (size_t) &( ((TYPE *)3264)->MEMBER ) - 3264 )
以上代码中没有使用 0 ,而是使用的 3264 ,所以绝对地址要减去3264才是该成员相对于结构体首地址的偏移地址,当然实际使用中我们都是用的是 0 。
转自:http://stackoverflow.com/questions/18554721/how-to-understand-size-t-type-0-member
相关文章推荐
- #define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
- #define offsetof(TYPE, MEMBER) (size_t)(&(((TYPE*)0)->MEMBER))
- #define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER);
- 如何解释 #define OFFSET(struct_type, member) ((size_t) &((struct_type *) 0)->member)
- #define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
- #define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
- vector<int>::size_type 解释
- vector<string>::size_type与string::size_type
- (size_t) & ((struct element *) 0) -> member
- std::vector<std::string>::size_type和 int的区别
- 理解vector<T>::size_type类型
- Windows7环境下简单安装Appache 2.2 和 PHP5(SSLSessionCache: Invalid argument: size has to be >= 8192 bytes )
- int( (LONG)(LONG_PTR)&(((type *)0)->field))的最好解释
- S动态添加与删除select中的Option对象 2009年09月23日 15:26 <html> <head> <meta http-equiv="Content-Type" content="t
- _BLOCK_TYPE_IS_VALID(pHead->nBlockUse
- spfile 、pfile、db启动顺序 SQL> show parameter db_recovery_file_dest_size; SQL> show parameter spfile
- 挂载LVM卷,提示mount: unknown filesystem type 'LVM2_member'的解决
- <typeAlias> <reslutmap> <parameterMap>
- error #2153: expression must have class type(低级错误:看是指针还是对象->和.的使用要正确)
- org.xml.sax.SAXParseException: Element type "beans" must be followed by either attribute specifications, ">" or "/>".