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

运行结果:


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值