整数0强制转化为指针的巧用
在工程中看到以下代码:
#define my_container_of(ptr,type,member) \
((type*)((char *) (ptr) - (unsigned long)(&((type*)0)->member)))
->的优先级高于&。 因此 &((type*)0)->member)的解释为先进行:
(type*)0)->member 取得member成员的运算,然后再进行取地址& 运算
举例测试:
#include <stdio.h>
int main(void)
{
typedef struct{
int a;
char b;
}Msg;
Msg my_var = {0};
printf("addres of my_var = 0x%x\n",&my_var);
printf("(unsigned long)(&((Msg*)0)->b) = 0x%x\n",(unsigned long)(&((Msg*)0)->b));
printf("address = 0x%x\n",(Msg*) ((char*)(&(my_var.b)) - (unsigned long)(&((Msg*)0)->b)));
printf("address of a = 0x%x\n",&(my_var.a));
return 0;
}
得到如下结果:
总结:
通过上述例子了解该代码目的:
1、(unsigned long)(&((Msg*)0)->b))获取结构体成员b 相对于结构体地址首地址的偏移量,然后通过获取自定义结构体变量my_var 中成员b的地址,减去b 相对于结构体首地址的偏移量,从而获取到结构体my_var的首地址。
应用:
struct my_slist_node
{
struct my_slist_node *next;
};
typedef struct my_slist_node my_slist_t; /**< single list. */
struct my_struct
{
const char *name;
uint32_t consumer_num;
osMutexId_t lock;
my_slist_t consumer_slist;
my_slist_t slist;
};
static my_slist_t my_root = {0};
如果my_root为首节点的 链表中保存着每个my_struct 变量中成员变量 slist 的地址,这样就可以通过该方法获取到每个my_struct 结构体变量的首地址,从而访问其结构体成员。