(size_t)&(((s *)0)->m)的理解
来源:互联网 发布:发论文数据可以造假吗 编辑:程序博客网 时间:2024/05/16 12:59
#define offsetof(s,m) (size_t)&(((s *)0)->m)
看了半天没弄懂,空指针怎么会不出错的,他们说没有写入操作,即没有 mov x, dword ptr []。找了下别人的理解,最详细的如下:
((s *)0):强制转化成数据结构指针,并使其指向地址0;
((s *)0)->m:使该指针指向成员m
&(((s *)0)->m):获取该成员m的地址
(size_t)&(((s *)0)->m):转化这个地址为合适的类型
你可能会迷惑,这样强制转换后的结构指针怎么可以用来访问结构体字段?呵呵,其实这个表达式根本没有也不打算访问m字段。ANSI C标准允许任何值为0的常量被强制转换成任何一种类型的指针,并且转换结果是一个NULL指针,因此((s*)0)的结果就是一个类型为s*的NULL指针。如果利用这个NULL指针来访问s的成员当然是非法的,但&(((s*)0)->m)的意图并非想存取s字段内容,而仅仅是计算当结构体实例的首址为((s*)0)时m字段的地址。聪明的编译器根本就不生成访问m的代码,而仅仅是根据s的内存布局和结构体实例首址在编译期计算这个(常量)地址,这样就完全避免了通过NULL指针访问内存的问题。又因为首址的值为0,所以这个地址的值就是字段相对于结构体基址的偏移。
这里有个地方需要注意:就是offsetof虽然同样适用于union结构,但它不能用于计算位域(bitfield)成员在数据结构中的偏移量。
typedef struct
{
unsigned int a:3;
unsigned int b:13;
unsigned int c:16;
}foo;
使用offset(foo,a)计算a在foo中的偏移量,编译器会报
本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/hzh007hzh/archive/2009/05/29/4224223.aspx
- (size_t)&(((s *)0)->m)的理解
- (size_t)&(((s *)0)->m)的理解 .
- (size_t)&(((s*)0)->m) 的理解
- #define offsetof(s,m) (size_t)&(((s *)0)->m)
- #define offsetof(s,m) (size_t)&(((s *)0)->m)
- while(cin>>str)无法结束的问题(zz)
- 重载">>"时遇到的错误error C2679
- >/dev/null 2>&1的含义
- ubuntu -->root用户的设定
- #define OFFSET(struc, m) (size_t)(&(((struc*)0)->m))
- 1>s.cpp(465) : error C2448: “main”: 函数样式初始值设定项类似函数定义 问题的解决方法
- (转) 进程 > 线程 > 消息队列,三者的包含关系
- 关于>&2、2>&1等重定向的解释[转载…
- 推荐>>黑客基地
- C#基础学习日志===>命名空间的使用<===
- >看书笔记~
- 傻->天使
- >--练习总结
- MFC ActiveX 控件在IE下闪烁的问题
- 因为你不喜欢我
- 计算页数方法
- bit操作的问题
- SQL Server索引管理之六大铁律
- (size_t)&(((s *)0)->m)的理解
- Java解决各种中文乱码问题(1)get和post请求
- 生成测试数据-3-T-SQL系列
- 尽量用库函数
- 汇编语言源程序的书写格式
- 用CSS控制浏览器滚动条样式源代码
- 二维数组 & 二级指针
- 标识符(函数名、变量名)的选择
- 大小写转化