c++根据结构体成员指针找到结构体指针

宏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 
#include 

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;
}

运行结果:

c++根据结构体成员指针找到结构体指针_第1张图片

你可能感兴趣的:(c++根据结构体成员指针找到结构体指针)