#define CONTAINING_RECORD(address, type, field) ((type *)( \
(PCHAR)(address) - \
(ULONG_PTR)(&((type *)0)->field)))
那么我们拆开来看,前半部分——(PCHAR)(address)不很难理解吧,获取了成员变量的地址,关键点在后半部分那长长的一条——(ULONG_PTR)(&((type *)0)->field),一点点看,先看最里面的那三对括号——(&((type *)0)->field),首先看((type *)0),这句话,其实是得到了指向当0为type类型结构体的首地址时的field成员变量指针。那么再加上&这个取地址符,就是field成员变量到结构体首地址的偏移量。
这样的话:成员变量的地址-成员变量和结构体首地址间的偏移量,就是结构体的首地址了,蛮巧妙地应用。