获取一个结构体中的某字段的偏移量,linux的内核代码中是这样定义实现的:
offset = (unsigned long)(&((type*)0))->member); 其中type就是结构体类型,member即某字段名。
这里简单的说明一下这句代码的意思,本来(type*)0就是一个空指针,使用空指针引用成员肯定会造成段错误,但是重点就是前门的
&这个符号,&符号表示我们只是取那个字段的地址,而不是要去引用这个字段的内容,所以不会造成段错误。
另外结构体中的字段在内存中分配的地址是连续的,当我们去读取某个字段时,就比如&((type*)0)->member,会先取结构体类型
type的地址,然后再计算member字段的偏移量(offset),再将type的地址加上member字段的偏移量,得到member字段真正的地
址,而这个动作是编译器帮我们完成的。
那么利用这个特性,我们就有办法得到member的偏移量。比如&ptr->member取member的地址,所以如果ptr的地址是100的话,那
么member的地址就是100+offset。那如果ptr是0的话,这样得到的结果就是offset,也就是member在type结构体中的偏移量。
最后而且利用这个特性,当知道member的地址时,我们也很容易得到结构体的地址值,即p = (type*)((char*)(ptr)-(unsigned long)(&((type*)0)->member))。