宏CONTAINING_RECORD,可以直接根据结构体成员指针找到结构体指针。
我们看一下它的定义:
#define CONTAINING_RECORD(addr,type,field) ((type*)((unsigned char*)addr - (unsigned long)&((type*)0)->field))
// addr: 结构体中某个成员变量的地址
// type: 结构体的原型
// field: 结构体的某个成员(与前面相同)
为什么这样就能得到结构体指针?
这个得益于结构体在申请内存空间是一段连续的地址,我们可以通过其中某个成员变量的地址,减去该成员的偏移来得到该成员所在结构体的指针。
那么为何(unsigned long)&((type*)0)->field))能得到某结构体成员的偏移?
这里可以假设某结构体指针地址是0000000,那么该成员的地址就是该成员对于其结构体的偏移
为了论证我们的猜想是对的,你可以无限次数运行以下代码
#include <windows.h>
#include <stdio.h>
struct T{
int a;
int b;
int c;
};
int main()
{
T t = {1,2,3};
printf("结构体指针地址:\t%p\n", &t );
printf("结构体成员a指针地址:\t%p\n", &t.a );
printf("结构体成员b指针地址:\t%p\n", &t.b );
printf("结构体成员c指针地址:\t%p\n", &t.c );
printf("...\n");
printf("下面利用结构体成员c来推算结构体的指针\n");
int *cc = &t.c;
T *tt = CONTAINING_RECORD(cc, T, c);
printf("结构体(T*)0指针地址:\t%p\n", ((T*)0) );
printf("结构体成员c的偏移值:\t%d\n", &((T*)0)->c );
printf("结构体指针地址:\t%p\n", tt );
system("pause");
return 0;
}
运行结果: